aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorFranz Zotter <fzotter@users.sourceforge.net>2012-08-21 08:10:42 +0000
committerFranz Zotter <fzotter@users.sourceforge.net>2012-08-21 08:10:42 +0000
commit5086631c287a954bae9c2bb4a987cb7f1eda3e57 (patch)
tree344393f7d6b8cbed979444c48f23003a232eb9c4 /src
parent585ca79e105c6e8ca57a2320c2335623a184ae06 (diff)
convex hull algorithm added
svn path=/trunk/externals/iem/iemmatrix/; revision=16165
Diffstat (limited to 'src')
-rw-r--r--src/iemmatrix_sources.c1
-rw-r--r--src/iemmatrix_sources.h1
-rw-r--r--src/mtx_qhull.c167
-rw-r--r--src/mtx_qhull/list.c294
-rw-r--r--src/mtx_qhull/list.h38
-rwxr-xr-xsrc/mtx_qhull/mtx_xprod.pd32
-rw-r--r--src/mtx_qhull/test.obj72
-rw-r--r--src/mtx_qhull/test_list.c94
-rw-r--r--src/mtx_qhull/test_write_conv_hull_obj.pd201
-rw-r--r--src/mtx_qhull/vectors.c126
-rw-r--r--src/mtx_qhull/vectors.h40
-rw-r--r--src/mtx_qhull/zhull.c482
-rw-r--r--src/mtx_qhull/zhull.h25
13 files changed, 1573 insertions, 0 deletions
diff --git a/src/iemmatrix_sources.c b/src/iemmatrix_sources.c
index 11bb4d4..0c739b6 100644
--- a/src/iemmatrix_sources.c
+++ b/src/iemmatrix_sources.c
@@ -72,6 +72,7 @@ void iemmatrix_sources_setup(void)
iemtx_powtodb_setup(); /* mtx_powtodb.c */
iemtx_print_setup(); /* mtx_print.c */
iemtx_prod_setup(); /* mtx_prod.c */
+ iemtx_qhull_setup(); /* mtx_qhull.c */
iemtx_qr_setup(); /* mtx_qr.c */
iemtx_rand_setup(); /* mtx_rand.c */
iemtx_repmat_setup(); /* mtx_repmat.c */
diff --git a/src/iemmatrix_sources.h b/src/iemmatrix_sources.h
index 0e24529..872be3a 100644
--- a/src/iemmatrix_sources.h
+++ b/src/iemmatrix_sources.h
@@ -70,6 +70,7 @@ void iemtx_pow_setup(void); /* mtx_pow.c */
void iemtx_powtodb_setup(void); /* mtx_powtodb.c */
void iemtx_print_setup(void); /* mtx_print.c */
void iemtx_prod_setup(void); /* mtx_prod.c */
+void iemtx_qhull_setup(void); /* mtx_qhull.c */
void iemtx_qr_setup(void); /* mtx_qr.c */
void iemtx_rand_setup(void); /* mtx_rand.c */
void iemtx_repmat_setup(void); /* mtx_repmat.c */
diff --git a/src/mtx_qhull.c b/src/mtx_qhull.c
new file mode 100644
index 0000000..700821f
--- /dev/null
+++ b/src/mtx_qhull.c
@@ -0,0 +1,167 @@
+/*
+ * iemmatrix
+ *
+ * objects for manipulating simple matrices
+ * own qhull algorithm implementation
+ *
+ * Copyright (c) 2012, Franz Zotter
+ * IEM, Graz, Austria
+ *
+ * For information on usage and redistribution, and for a DISCLAIMER OF ALL
+ * WARRANTIES, see the file, "LICENSE.txt," in this distribution.
+ *
+ */
+
+
+#include "iemmatrix.h"
+#include <stdlib.h>
+#include "mtx_qhull/list.h"
+#include "mtx_qhull/vectors.h"
+#include "mtx_qhull/zhull.h"
+#include "mtx_qhull/list.c"
+#include "mtx_qhull/vectors.c"
+#include "mtx_qhull/zhull.c"
+
+static t_class *mtx_qhull_class;
+
+typedef struct MTXQhull_ MTXQhull;
+struct MTXQhull_ {
+ t_object x_obj;
+ t_outlet *outl;
+ t_atom *list;
+ size_t size;
+ size_t hull_size;
+ int iter;
+ zhull_t *zh;
+};
+
+static void deleteMTXQhull(MTXQhull *xo) {
+ if (xo->zh!=0) {
+ free(xo->zh);
+ xo->zh=0;
+ }
+ if (xo->list!=0) {
+ free(xo->list);
+ xo->list=0;
+ xo->size=0;
+ }
+}
+
+void mTXQhullSetIterations(MTXQhull *xo, t_float f) {
+ xo->iter=(int)f;
+ if (xo->iter<0)
+ xo->iter=0;
+}
+
+static void *newMTXQhull(t_symbol *s, int argc, t_atom *argv) {
+ int nmax;
+ MTXQhull *xo = (MTXQhull *) pd_new (mtx_qhull_class);
+ xo->outl = outlet_new (&xo->x_obj, gensym("matrix"));
+ xo->zh=0;
+ xo->list=0;
+ xo->size=0;
+ xo->iter=0;
+ if (argc>0)
+ mTXQhullSetIterations(xo,atom_getfloat(argv));
+ return ((void *) xo);
+}
+
+
+static void mTXQhullMatrix(MTXQhull *xo, t_symbol *s,
+ int argc, t_atom *argv) {
+ int i;
+ int rows=atom_getint(argv++);
+ int columns=atom_getint(argv++);
+ int size=rows*columns;
+ int in_size=argc-2;
+ float *x;
+ float *y;
+ float *z;
+
+ /* size check */
+ if (!size)
+ post("mtx_qhull: invalid dimensions");
+ else if (in_size<size)
+ post("mtx_qhull: sparse matrix not yet supported: use \"mtx_check\"");
+ else if ((rows<4)||(columns!=3))
+ post("mtx_qhull: requires an L x 3 matrix with at least L>=4");
+ else {
+ xo->zh = (zhull_t*)malloc(sizeof(zhull_t));
+ x=(float*)malloc(rows*sizeof(float));
+ y=(float*)malloc(rows*sizeof(float));
+ z=(float*)malloc(rows*sizeof(float));
+ if ((x!=0)&&(y!=0)&&(z!=0)&&(xo->zh!=0)) {
+ for (i=0; i<rows; i++) {
+ x[i]=atom_getfloat(argv++);
+ y[i]=atom_getfloat(argv++);
+ z[i]=atom_getfloat(argv++);
+ }
+ *(xo->zh)=zhullInitPoints(x,y,z,rows);
+ calculateZHull(xo->zh,xo->iter);
+ free(x);
+ free(y);
+ free(z);
+ xo->hull_size=getLength(xo->zh->facets);
+ if (xo->list==0) {
+ xo->list = (t_atom*)malloc((xo->hull_size*3+2)*sizeof(t_atom));
+ } else {
+ xo->list = (t_atom*)realloc(xo->list,(xo->hull_size*3+2)*sizeof(t_atom));
+ }
+ if(xo->list!=0) {
+ xo->size=(xo->hull_size*3+2);
+ SETFLOAT(xo->list,(float)xo->hull_size);
+ SETFLOAT(xo->list+1,(float)3);
+ for (i=0; i<xo->hull_size; i++) {
+ SETFLOAT(xo->list+2+3*i, (float)getEntry(
+ getFacetByIndex(xo->zh->facets,i)->corners,0)+1);
+ SETFLOAT(xo->list+3+3*i, (float)getEntry(
+ getFacetByIndex(xo->zh->facets,i)->corners,1)+1);
+ SETFLOAT(xo->list+4+3*i, (float)getEntry(
+ getFacetByIndex(xo->zh->facets,i)->corners,2)+1);
+ }
+ outlet_anything(xo->outl, gensym("matrix"),
+ xo->size, xo->list);
+ freeZhull(xo->zh);
+ free(xo->zh);
+ xo->zh=0;
+ }
+ else {
+ post("mtx_qhull: memory problem, no operation!");
+ xo->size=0;
+ freeZhull(xo->zh);
+ free(xo->zh);
+ xo->zh=0;
+ }
+ }
+ else {
+ if(x!=0)
+ free(x);
+ if(y!=0)
+ free(y);
+ if(z!=0)
+ free(z);
+ if(xo->zh!=0)
+ free(xo->zh);
+ xo->zh=0;
+ post("mtx_qhull: memory error, no operation!");
+ }
+
+ }
+}
+
+void mtx_qhull_setup (void) {
+ mtx_qhull_class = class_new (
+ gensym("mtx_qhull"),
+ (t_newmethod) newMTXQhull,
+ (t_method) deleteMTXQhull,
+ sizeof(MTXQhull),
+ CLASS_DEFAULT, A_GIMME, 0);
+ class_addmethod(mtx_qhull_class, (t_method) mTXQhullMatrix, gensym("matrix"), A_GIMME, 0);
+ class_addfloat(mtx_qhull_class, (t_method) mTXQhullSetIterations);
+}
+
+void iemtx_qhull_setup(void){
+ mtx_qhull_setup();
+}
+
+
diff --git a/src/mtx_qhull/list.c b/src/mtx_qhull/list.c
new file mode 100644
index 0000000..07ce4f2
--- /dev/null
+++ b/src/mtx_qhull/list.c
@@ -0,0 +1,294 @@
+
+// memory things:
+list_t emptyList(void) {
+ list_t generated_list;
+ generated_list.length=0;
+ generated_list.entries=0;
+ return generated_list;
+}
+
+list_t allocateList(const size_t length) {
+ list_t generated_list = emptyList();
+ if (length>0) {
+ generated_list.entries= (entry_t*) malloc(length*sizeof(entry_t));
+ if (generated_list.entries!=0) {
+ generated_list.length=length;
+ // printf("got list %li\n",generated_list.entries);
+ }
+ }
+ return generated_list;
+}
+
+void reallocateList(list_t *list,
+ const size_t length) {
+ entry_t *old_entries = list->entries;
+ if (length>0) {
+ if (list->length==0)
+ *list = allocateList(length);
+ else {
+ if (list->length != length)
+ list->entries = (entry_t*) realloc(list->entries,length*sizeof(entry_t));
+ if (list->entries!=0)
+ list->length=length;
+ else
+ *list=emptyList();
+ }
+ }
+ else
+ freeList(list);
+ /* if ((list->entries!=old_entries)&&(old_entries!=0)) {
+ if (list->entries!=0)
+ printf("moved %li by realloc to %li\n",old_entries,list->entries);
+ else
+ printf("freed %li by realloc\n", old_entries);
+ }*/
+}
+
+
+void freeList(list_t *list) {
+ if (list->entries!=0) {
+ // printf("deleting list %li\n",list->entries);
+ free(list->entries);
+ }
+ list->entries=0;
+ list->length=0;
+}
+
+// programming interface:
+
+size_t getLength(const list_t list) {
+ return list.length;
+}
+
+entry_t getEntry(const list_t list, const index_t index) {
+ if ((index>=0)&&(index<getLength(list)))
+ return list.entries[index];
+ else
+ return 0;
+}
+
+void setEntry(const list_t list, const index_t index, const entry_t entry) {
+ if ((index>=0)&&(index<getLength(list)))
+ list.entries[index]=entry;
+}
+
+
+list_t initList(entry_t *entries, const size_t length) {
+ index_t i;
+ list_t l = allocateList(length);
+ if (l.entries!=0)
+ for (i=0; i<(index_t)length; i++)
+ setEntry(l,i,entries[i]);
+ return l;
+}
+
+list_t initListFromTo(const entry_t start, const entry_t stop) {
+ index_t i;
+ size_t length;
+ entry_t c;
+ int incr;
+ if (stop>=start) {
+ length=(size_t) (stop-start+1);
+ incr=1;
+ } else {
+ length=(size_t) (start-stop+1);
+ incr=-1;
+ }
+ list_t l = allocateList(length);
+ if (l.entries!=0)
+ for (i=0,c=start; i<length; i++, c+=incr)
+ setEntry(l,i,c);
+ return l;
+}
+
+list_t initConstantList(const entry_t c, const size_t length){
+ entry_t i;
+ list_t l = allocateList(length);
+ if (l.entries!=0)
+ for (i=0; i<length; i++)
+ setEntry(l,i,c);
+ return l;
+}
+
+
+list_t duplicateList(const list_t list_in) {
+ entry_t i;
+ list_t list_out=emptyList();
+ list_out = allocateList(getLength(list_in));
+ for (i=0; i<getLength(list_out); i++)
+ setEntry(list_out,i,getEntry(list_in,i));
+ return list_out;
+}
+
+list_t mergeLists(const list_t list1, const list_t list2) {
+ list_t list_out;
+ entry_t i,j;
+ list_out = allocateList(getLength(list1)+ getLength(list2));
+ if (list_out.entries!=0) {
+ for (i=0; i<getLength(list1); i++)
+ setEntry(list_out,i,getEntry(list1,i));
+ for (j=0; i<getLength(list_out); i++, j++)
+ setEntry(list_out,i,getEntry(list2,j));
+ }
+ return list_out;
+}
+
+list_t getSubList(const list_t list, const list_t indices) {
+ index_t i;
+ list_t new_list = allocateList(getLength(indices));
+ for (i=0; i<getLength(new_list); i++) {
+ setEntry(new_list,i,getEntry(list,getEntry(indices,i)));
+ }
+}
+
+list_t getSubListFromTo(const list_t list, const index_t start,
+ const index_t stop) {
+ list_t new_list=emptyList();
+ index_t i,j;
+ int incr;
+ if ((start>0)&&(stop>0)&&(start<getLength(list))&&(stop<getLength(list))) {
+ if (start>stop) {
+ incr=-1;
+ new_list=allocateList(start-stop+1);
+ } else {
+ incr=1;
+ new_list=allocateList(start-stop+1);
+ }
+ for (j=start,i=0; i<getLength(new_list); i++, j+=incr) {
+ setEntry(new_list,i,getEntry(list,j));
+ }
+ }
+ return new_list;
+}
+
+void appendToList(list_t *list, const entry_t entry) {
+ const index_t i=(index_t)getLength(*list);
+ reallocateList(list,getLength(*list)+1);
+ if (getLength(*list)>(size_t)i) {
+ setEntry(*list,i,entry);
+ }
+}
+
+void removeEntryFromList(list_t *list, const index_t index) {
+ index_t i,j;
+ for (i=j=0; i<getLength(*list); i++) {
+ if (i!=index)
+ setEntry(*list,j++,getEntry(*list,i));
+ }
+ reallocateList(list,j);
+}
+
+
+void removeValueFromList(list_t *list, const entry_t entry) {
+ index_t i,j;
+ for (j=i=0; i<getLength(*list); i++) {
+ if (getEntry(*list,i)!=entry)
+ setEntry(*list, j++, getEntry(*list,i));
+ }
+ reallocateList(list,j);
+}
+
+void appendListToList(list_t *list1, const list_t list2) {
+ index_t i,j;
+ size_t siz_old = getLength(*list1);
+ siz_old=list1->length;
+ reallocateList(list1, getLength(*list1) + getLength(list2));
+ if (getLength(*list1)>siz_old) {
+ for (i=siz_old, j=0; i<list1->length; i++, j++)
+ setEntry(*list1,i,getEntry(list2,j));
+ }
+}
+
+void removeEntryListFromList(list_t *list, const list_t indices) {
+ index_t i,j;
+ for (i=j=0; i<getLength(*list); i++)
+ if (notInList(i,indices))
+ setEntry(*list, j++, getEntry(*list,i));
+ reallocateList(list,j);
+}
+
+void removeValueListFromList(list_t *list, const list_t excl_list) {
+ index_t i,j,k;
+ int keep;
+ for (j=i=0; i<getLength(*list); i++) {
+ keep=1;
+ for (k=0; k<getLength(excl_list); k++)
+ keep=(keep)&&(getEntry(*list,i)!=getEntry(excl_list,k));
+ if (keep)
+ setEntry(*list, j++, getEntry(*list,i));
+ }
+ reallocateList(list,j);
+}
+
+void reverseList(list_t * const list) {
+ index_t i,j;
+ entry_t v;
+ const cnt = getLength(*list)/ 2;
+ if (cnt>0)
+ for (i=0, j=getLength(*list)-1; i<cnt; i++, j--) {
+ v=getEntry(*list,i);
+ setEntry(*list,i,getEntry(*list,j));
+ setEntry(*list,j,v);
+ }
+}
+
+int inList(const entry_t entry, const list_t list) {
+ index_t i;
+ for (i=0; i<getLength(list); i++)
+ if (getEntry(list,i)==entry)
+ return 1;
+ return 0;
+}
+
+entry_t findValueInList(const entry_t entry, const list_t list) {
+ index_t i;
+ for (i=0; i<getLength(list); i++)
+ if (entry==getEntry(list,i))
+ return i;
+ return getLength(list);
+}
+
+void uniquefyListEntries(list_t *list) {
+ index_t i,j,k;
+ int keep;
+ k=0;
+ for (j=0; j<getLength(*list); j++) {
+ keep=1;
+ for (i=0; (i<k)&&(keep); i++)
+ keep = (keep)&&(list->entries[j]!=list->entries[i]);
+ if (keep) {
+ list->entries[i++]=list->entries[j];
+ k++;
+ }
+ }
+ reallocateList(list, k);
+}
+
+list_t findValueListInList(const list_t value_list,
+ const list_t list) {
+ list_t l=emptyList();
+ index_t i,j;
+ for (i=0; i<getLength(value_list); i++)
+ appendToList(&l,findValueInList(
+ getEntry(value_list,i),list));
+ return l;
+}
+
+int notInList(const entry_t entry, const list_t list) {
+ index_t i;
+ for (i=0; i<getLength(list); i++)
+ if (getEntry(list,i)==entry)
+ return 0;
+ return 1;
+}
+
+void printList(list_t const list) {
+ entry_t i;
+ printf("[list]_%d=[",list.length);
+ if (getLength(list)>0)
+ printf("%d",getEntry(list,0));
+ for (i=1; i<list.length; i++)
+ printf(", %d",getEntry(list,i));
+ printf("]\n");
+}
+
diff --git a/src/mtx_qhull/list.h b/src/mtx_qhull/list.h
new file mode 100644
index 0000000..699f9ed
--- /dev/null
+++ b/src/mtx_qhull/list.h
@@ -0,0 +1,38 @@
+
+typedef long int entry_t;
+typedef long int index_t;
+
+typedef struct list_ {
+ entry_t *entries;
+ size_t length;
+} list_t;
+
+// memory things:
+list_t emptyList(void);
+void freeList(list_t *list);
+
+// programming interface:
+size_t getLength(const list_t list);
+entry_t getEntry(const list_t list, const index_t index);
+void setEntry(const list_t list, const index_t index, const entry_t entry);
+list_t initList(entry_t *entries, const size_t length);
+list_t initListFromTo(const entry_t start, const entry_t stop);
+list_t initConstantList(const entry_t c, const size_t length);
+list_t duplicateList(const list_t list_in);
+list_t mergeLists(const list_t list1, const list_t list2);
+list_t getSubList(const list_t list, const list_t indices);
+list_t getSubListFromTo(const list_t list, const index_t start,
+ const index_t stop);
+void appendToList(list_t *list, const entry_t entry);
+void removeValueFromList(list_t *list, const entry_t entry);
+void removeEntryFromList(list_t *list, const index_t index);
+void appendListToList(list_t *list1, const list_t list2);
+void removeValueListFromList(list_t *list, const list_t excl_list);
+void removeEntryListFromList(list_t *list, const list_t indices);
+void reverseList(list_t * const list);
+int inList(const entry_t entry, const list_t list);
+int notInList(const entry_t entry, const list_t list);
+list_t findValueListInList(const list_t value_list, const list_t list);
+entry_t findValueInList(const entry_t entry, const list_t list);
+void printList(const list_t list);
+
diff --git a/src/mtx_qhull/mtx_xprod.pd b/src/mtx_qhull/mtx_xprod.pd
new file mode 100755
index 0000000..08ac92c
--- /dev/null
+++ b/src/mtx_qhull/mtx_xprod.pd
@@ -0,0 +1,32 @@
+#N canvas 502 62 631 420 12;
+#X obj 73 53 inlet;
+#X obj 473 54 inlet;
+#X obj 73 81 t a a;
+#X obj 73 202 mtx_.*;
+#X obj 197 201 mtx_.*;
+#X obj 73 239 mtx_-;
+#X obj 73 287 outlet;
+#X text 180 34 mtx: cross product;
+#X text 291 206 matrix1: 3xN;
+#X text 291 227 matrix2: 3xN;
+#X text 291 252 ============;
+#X text 299 273 matrix: 3xN;
+#X text 407 206 [a1 a2 a3 ...];
+#X text 408 227 [b1 b2 b3 ...];
+#X text 410 272 [a1 x b1 \, a2 x b2 \, ...];
+#X obj 73 114 mtx_scroll -1;
+#X obj 197 114 mtx_scroll 1;
+#X obj 345 112 mtx_scroll 1;
+#X obj 473 113 mtx_scroll -1;
+#X connect 0 0 2 0;
+#X connect 1 0 17 0;
+#X connect 1 0 18 0;
+#X connect 2 0 15 0;
+#X connect 2 1 16 0;
+#X connect 3 0 5 0;
+#X connect 4 0 5 1;
+#X connect 5 0 6 0;
+#X connect 15 0 3 0;
+#X connect 16 0 4 0;
+#X connect 17 0 3 1;
+#X connect 18 0 4 1;
diff --git a/src/mtx_qhull/test.obj b/src/mtx_qhull/test.obj
new file mode 100644
index 0000000..4b4e602
--- /dev/null
+++ b/src/mtx_qhull/test.obj
@@ -0,0 +1,72 @@
+v 1.5 0 0
+v 1.37304 0.603839 0.0119395
+v 1.00025 1.11771 0.0145916
+v 0.448097 1.4314 0.0170812
+v -0.339153 1.46107 0.0157868
+v -1.12324 0.994132 0.00513365
+v -1.49996 0.00517674 0.00970638
+v -1.1198 -0.99802 0.00258554
+v -0.284272 -1.47273 0.0159703
+v 0.518517 -1.40748 0.0113897
+v 1.06359 -1.05766 0.0118678
+v 1.39616 -0.548269 0.0118545
+v 1.21634 0.507813 0.715999
+v 0.496716 1.22144 0.715096
+v -0.543141 1.2091 0.702188
+v -1.21884 0.496413 0.719732
+v -1.20562 -0.527837 0.719625
+v -0.520718 -1.21172 0.714562
+v 0.548338 -1.19922 0.714976
+v 1.22223 -0.511408 0.703294
+v 0.558944 0.5949 1.25844
+v -0.560745 0.592992 1.25854
+v -0.566714 -0.600261 1.25241
+v 0.581676 -0.550909 1.26813
+f 24 23 22
+f 24 22 21
+f 19 23 24
+f 24 20 19
+f 23 19 18
+f 23 18 17
+f 16 22 23
+f 23 17 16
+f 22 16 15
+f 14 21 22
+f 22 15 14
+f 24 21 13
+f 13 20 24
+f 21 14 13
+f 11 19 20
+f 20 12 11
+f 19 11 10
+f 9 18 19
+f 19 10 9
+f 8 17 18
+f 18 9 8
+f 8 9 10
+f 7 16 17
+f 17 8 7
+f 6 15 16
+f 16 7 6
+f 6 7 8
+f 5 14 15
+f 15 6 5
+f 14 5 4
+f 4 5 6
+f 3 13 14
+f 3 5 6
+f 14 4 3
+f 3 4 6
+f 13 3 2
+f 20 13 1
+f 1 12 20
+f 1 11 12
+f 1 10 11
+f 1 8 10
+f 1 6 8
+f 1 5 6
+f 1 4 6
+f 1 3 6
+f 1 3 4
+f 13 2 1
+f 1 2 3
diff --git a/src/mtx_qhull/test_list.c b/src/mtx_qhull/test_list.c
new file mode 100644
index 0000000..4ddddde
--- /dev/null
+++ b/src/mtx_qhull/test_list.c
@@ -0,0 +1,94 @@
+#include <stdio.h>
+#include <stdlib.h>
+#include "list.h"
+
+int main(char **argv, int argc) {
+ list_t l1=emptyList();
+ list_t l2=emptyList();
+ list_t l3=emptyList();
+ int x[]={0, 2, 4, 6};
+
+ printf("empty list:\n");
+ printList(l1);
+ freeList(&l1);
+
+ printf("constant list with 10 1 entries:\n");
+ l1=initConstantList(1,10);
+ printList(l1);
+ freeList(&l1);
+
+ printf("list from 2 to 3:\n");
+ l1=initListFromTo(2,3);
+ printList(l1);
+ printf("list from 7 to 1:\n");
+ l2=initListFromTo(7,1);
+ printList(l2);
+ printf("duplicate list from 2 to 3:\n");
+ l3=duplicateList(l1);
+ printList(l3);
+ freeList(&l3);
+
+ printf("merge list from 1 to 7 with list from 2 to 3:\n");
+ l3=mergeLists(l1,l2);
+ printList(l3);
+
+ printf("remove entry 6 list from list:\n");
+ removeValueFromList(&l3,6);
+ printList(l3);
+
+ printf("remove entries [2, 3] from list:\n");
+ removeValueListFromList(&l3,l1);
+ printList(l3);
+
+ printf("reverse list:\n");
+ reverseList (&l3);
+ printList(l3);
+
+ printf("append entry 8 to list:\n");
+ appendToList(&l3,8);
+ printList(l3);
+
+ printf("is 8 not in list?: %d\n",notInList(8,l3));
+ printf("is 3 not in list?: %d\n",notInList(3,l3));
+
+ printf("remove item 4 from list\n");
+ removeEntryFromList(&l3,4);
+ printList(l3);
+
+ printf("remove item 1 from list\n");
+ removeEntryFromList(&l3,1);
+ printList(l3);
+
+ printf("remove item 0 from list\n");
+ removeEntryFromList(&l3,0);
+ printList(l3);
+
+ printf("...taking a longer list\n");
+ printList(l2);
+ freeList(&l3);
+ l3=initList(x,4);
+ printf("removing index list ");
+ printList(l3);
+ removeEntryListFromList(&l2,l3);
+ printList(l2);
+ freeList(&l1);
+ freeList(&l2);
+ freeList(&l3);
+
+ l1=initListFromTo(1,3);
+ l2=initListFromTo(0,5);
+ printf("...taking a longer list\n");
+ printList(l1);
+ printf("finding indices of values in list");
+ printList(l2);
+ l3=findValueListInList(l2,l1);
+ printList(l3);
+
+
+ freeList(&l1);
+ freeList(&l2);
+ freeList(&l3);
+
+}
+
+// gcc list.c test_list.c && ./a.out
diff --git a/src/mtx_qhull/test_write_conv_hull_obj.pd b/src/mtx_qhull/test_write_conv_hull_obj.pd
new file mode 100644
index 0000000..d27370d
--- /dev/null
+++ b/src/mtx_qhull/test_write_conv_hull_obj.pd
@@ -0,0 +1,201 @@
+#N canvas 534 0 850 668 10;
+#X obj 595 205 convhull 0.001;
+#X obj 595 176 t a a;
+#N canvas 69 11 539 331 writeobject 0;
+#X obj 163 270 textfile;
+#X obj 163 161 mtx;
+#X obj 163 123 until;
+#X obj 200 123 f;
+#X obj 226 123 + 1;
+#X msg 225 102 1;
+#X obj 163 81 t a a b;
+#X obj 163 102 mtx_size;
+#X obj 143 32 inlet;
+#X obj 373 44 inlet;
+#X msg 199 142 row \$1;
+#X obj 163 238 list trim;
+#X msg 269 196 add \$1;
+#X msg 329 174 symbol v;
+#X obj 373 65 t a b;
+#X obj 143 54 t b a b;
+#X msg 45 230 write test.obj cr;
+#X msg 399 108 clear;
+#X obj 164 218 list prepend;
+#X msg 269 174 symbol f;
+#X connect 1 0 18 0;
+#X connect 2 0 3 0;
+#X connect 3 0 4 0;
+#X connect 3 0 10 0;
+#X connect 4 0 3 1;
+#X connect 5 0 3 1;
+#X connect 6 0 7 0;
+#X connect 6 1 1 1;
+#X connect 6 2 5 0;
+#X connect 7 0 2 0;
+#X connect 8 0 15 0;
+#X connect 9 0 14 0;
+#X connect 10 0 1 0;
+#X connect 11 0 0 0;
+#X connect 12 0 18 1;
+#X connect 13 0 12 0;
+#X connect 14 0 6 0;
+#X connect 14 1 13 0;
+#X connect 14 1 17 0;
+#X connect 15 0 16 0;
+#X connect 15 1 6 0;
+#X connect 15 2 19 0;
+#X connect 16 0 0 0;
+#X connect 17 0 0 0;
+#X connect 18 0 11 0;
+#X connect 19 0 12 0;
+#X restore 116 490 pd writeobject;
+#N canvas 0 22 450 300 gemwin 0;
+#X obj 132 136 gemwin;
+#X obj 67 89 outlet;
+#X obj 67 10 inlet;
+#X obj 67 41 route create;
+#X msg 67 70 set destroy;
+#X msg 142 68 set create;
+#X msg 132 112 create \, 1 \, lighting 1;
+#X msg 298 112 destroy \, reset;
+#X msg 256 79 color 0.78 0.8 1;
+#X obj 256 57 loadbang;
+#X connect 2 0 3 0;
+#X connect 3 0 4 0;
+#X connect 3 0 6 0;
+#X connect 3 1 5 0;
+#X connect 3 1 7 0;
+#X connect 4 0 1 0;
+#X connect 5 0 1 0;
+#X connect 6 0 0 0;
+#X connect 7 0 0 0;
+#X connect 8 0 0 0;
+#X connect 9 0 8 0;
+#X restore 556 423 pd gemwin;
+#X msg 556 404 destroy;
+#X obj 668 404 gemhead;
+#X obj 668 423 world_light;
+#X obj 362 -1 gemhead;
+#X obj 393 151 model;
+#X floatatom 377 79 5 0 0 0 - - -;
+#X floatatom 420 80 5 0 0 0 - - -;
+#X floatatom 460 80 5 0 0 0 - - -;
+#X msg 17 196 open test.obj;
+#X msg 500 97 rescale 0;
+#X obj 364 38 color 0.9 0.9 0.9;
+#X obj 595 227 extra_point;
+#X obj 209 467 mtx_* 1.5;
+#X obj 363 18 ortho;
+#X obj 488 -3 tgl 15 0 empty empty empty 17 7 0 10 -262144 -1 -1 0
+1;
+#X obj 485 38 f;
+#X obj 540 38 % 360;
+#X obj 480 17 metro 50;
+#X msg 360 176 smooth 0;
+#X obj 595 266 convhull 0.001;
+#X obj 595 248 t a a;
+#X obj 85 411 mtx;
+#X obj 84 446 mtx_slice;
+#X msg 123 425 1 1 \$1 end;
+#X floatatom 18 250 5 0 0 0 - - -;
+#X obj 221 440 mtx;
+#X obj -7 284 t b b f b;
+#X obj 125 394 mtx_size;
+#X obj 101 375 t a a;
+#X msg 184 396 set \$1;
+#X obj 355 101 rotateXYZ -74 0 0;
+#X obj 513 38 + 3;
+#X obj 360 128 t b a;
+#X obj 113 332 convhull 0.001;
+#X obj 159 229 t a a;
+#X msg 105 72 4 40;
+#X msg 109 92 5 67;
+#X msg 103 50 3 26;
+#X msg 101 137 read designsN/N\$1_\$2.mtx \, bang;
+#X obj 60 135 mtx;
+#X obj 183 280 demux;
+#X obj 210 213 tgl 15 0 empty empty empty 17 7 0 10 -262144 -1 -1 1
+1;
+#X obj 249 237 t b f;
+#X obj 60 165 t b a b;
+#X msg 111 111 9 180;
+#X msg 97 1 1.5 6;
+#X msg 104 27 2 14;
+#X msg 181 43 2.5 12;
+#X floatatom 323 280 5 0 0 0 - - -;
+#X obj 327 301 t b f;
+#X obj 416 257 mtx_rand;
+#X msg 427 235 200 3;
+#X obj 245 336 mtx_qhull 500;
+#X msg 175 77 read designsN/dode.mtx \, bang;
+#X msg 221 167 read Rls.mtx \, bang;
+#X connect 0 0 15 0;
+#X connect 1 0 0 0;
+#X connect 1 1 15 1;
+#X connect 3 0 4 0;
+#X connect 4 0 3 0;
+#X connect 5 0 6 0;
+#X connect 7 0 17 0;
+#X connect 9 0 34 1;
+#X connect 10 0 34 2;
+#X connect 11 0 34 3;
+#X connect 12 0 36 0;
+#X connect 13 0 8 0;
+#X connect 14 0 34 0;
+#X connect 15 0 24 0;
+#X connect 16 0 2 1;
+#X connect 17 0 14 0;
+#X connect 18 0 21 0;
+#X connect 19 0 11 0;
+#X connect 19 0 35 0;
+#X connect 20 0 19 1;
+#X connect 21 0 19 0;
+#X connect 22 0 8 0;
+#X connect 23 0 32 0;
+#X connect 24 0 23 0;
+#X connect 24 1 29 0;
+#X connect 25 0 26 0;
+#X connect 26 0 2 0;
+#X connect 27 0 26 1;
+#X connect 28 0 30 0;
+#X connect 29 0 16 0;
+#X connect 30 0 12 0;
+#X connect 30 1 25 0;
+#X connect 30 2 27 0;
+#X connect 30 3 29 0;
+#X connect 31 0 27 0;
+#X connect 31 0 33 0;
+#X connect 32 0 25 0;
+#X connect 32 1 31 0;
+#X connect 33 0 28 0;
+#X connect 34 0 8 0;
+#X connect 35 0 20 0;
+#X connect 36 0 22 0;
+#X connect 36 1 8 0;
+#X connect 37 0 32 0;
+#X connect 38 0 44 0;
+#X connect 38 1 29 0;
+#X connect 39 0 42 0;
+#X connect 40 0 42 0;
+#X connect 41 0 42 0;
+#X connect 42 0 43 0;
+#X connect 43 0 47 0;
+#X connect 44 0 37 0;
+#X connect 44 1 56 0;
+#X connect 45 0 46 0;
+#X connect 46 0 43 0;
+#X connect 46 1 44 1;
+#X connect 47 0 12 0;
+#X connect 47 1 38 0;
+#X connect 48 0 42 0;
+#X connect 49 0 42 0;
+#X connect 50 0 42 0;
+#X connect 51 0 42 0;
+#X connect 52 0 53 0;
+#X connect 53 0 43 0;
+#X connect 53 1 56 0;
+#X connect 54 0 43 0;
+#X connect 55 0 54 0;
+#X connect 56 0 32 0;
+#X connect 57 0 43 0;
+#X connect 58 0 43 0;
diff --git a/src/mtx_qhull/vectors.c b/src/mtx_qhull/vectors.c
new file mode 100644
index 0000000..909543e
--- /dev/null
+++ b/src/mtx_qhull/vectors.c
@@ -0,0 +1,126 @@
+
+vector_t initVector (float x, float y, float z) {
+ vector_t vec={x, y, z};
+ return vec;
+}
+
+vector_t normalizeVector(vector_t v) {
+ float r=sqrtf(v.c[0]*v.c[0]+v.c[1]*v.c[1]+v.c[2]*v.c[2]);
+ v.c[0]/=r;
+ v.c[1]/=r;
+ v.c[2]/=r;
+ return v;
+};
+
+plane_t initPlane (vector_t normal, vector_t point) {
+ plane_t plane;
+ plane.point = point;
+ plane.normal = normalizeVector(normal);
+ return plane;
+}
+
+points_t initPoints (const float *x,
+ const float *y, const float *z,
+ size_t num_points) {
+ points_t points;
+ size_t i;
+
+ points.v = (vector_t*) malloc (sizeof(vector_t)*num_points);
+ if (points.v!=0) {
+// printf("created %li\n",points.v);
+ points.num_points = num_points;
+ for (i=0; i<num_points; i++) {
+ points.v[i] = initVector(x[i],y[i],z[i]);
+ }
+ }
+ return points;
+}
+
+void freePoints (points_t *points) {
+// printf("deleting %li\n",points->v);
+ if (points!=0) {
+ if (points->v!=0)
+ free(points->v);
+ points->v = 0;
+ points->num_points = 0;
+ }
+}
+
+vector_t crossProduct (vector_t v1, vector_t v2) {
+ vector_t cp;
+ cp.c[0]= v1.c[1]*v2.c[2]-v1.c[2]*v2.c[1];
+ cp.c[1]=-v1.c[0]*v2.c[2]+v1.c[2]*v2.c[0];
+ cp.c[2]= v1.c[0]*v2.c[1]-v1.c[1]*v2.c[0];
+ return cp;
+}
+
+float innerProduct (vector_t v1, vector_t v2) {
+ return v1.c[0]*v2.c[0] + v1.c[1]*v2.c[1] + v1.c[2]*v2.c[2];
+}
+
+float distancePointPlane (vector_t point, plane_t plane) {
+ return innerProduct(point, plane.normal) -
+ innerProduct(plane.point, plane.normal);
+}
+
+vector_t addVectors(vector_t v1, vector_t v2) {
+ vector_t v3;
+ v3.c[0]=v1.c[0]+v2.c[0];
+ v3.c[1]=v1.c[1]+v2.c[1];
+ v3.c[2]=v1.c[2]+v2.c[2];
+ return v3;
+}
+
+vector_t subtractVectors(vector_t v1, vector_t v2) {
+ vector_t v3;
+ v3.c[0]=v1.c[0]-v2.c[0];
+ v3.c[1]=v1.c[1]-v2.c[1];
+ v3.c[2]=v1.c[2]-v2.c[2];
+ return v3;
+}
+
+vector_t scaleVector(vector_t v1, float f) {
+ vector_t v2;
+ v2.c[0]=f*v1.c[0];
+ v2.c[1]=f*v1.c[1];
+ v2.c[2]=f*v1.c[2];
+ return v2;
+}
+
+vector_t averagePoints(points_t points) {
+ vector_t m = initVector(0.0f, 0.0f, 0.0f);
+ size_t i;
+ for (i=0; i<points.num_points; i++)
+ m=addVectors(points.v[i],m);
+ m=scaleVector(m,1.0f/((float)points.num_points));
+ return m;
+}
+
+vector_t normalOfPoints(points_t points) {
+ vector_t m = averagePoints(points);
+ vector_t n = initVector(0.0f, 0.0f, 0.0f);
+ size_t i;
+ for (i=0; i<points.num_points; i++) {
+ n=addVectors(crossProduct(points.v[i],m),n);
+ }
+ return n;
+}
+
+plane_t planeFromPoints(points_t points) {
+ vector_t p=averagePoints(points);
+ vector_t n=normalOfPoints(points);
+ plane_t pl=initPlane(n,p);
+ return pl;
+}
+
+void printPlane(plane_t p) {
+ printf("n=");
+ printVector(p.normal);
+ printf(", p=");
+ printVector(p.point);
+}
+
+void printVector(vector_t v) {
+ printf("[%5.2f,%5.2f,%5.2f], ", v.c[0],v.c[1],v.c[2]);
+}
+
diff --git a/src/mtx_qhull/vectors.h b/src/mtx_qhull/vectors.h
new file mode 100644
index 0000000..201205a
--- /dev/null
+++ b/src/mtx_qhull/vectors.h
@@ -0,0 +1,40 @@
+typedef struct vec_ {
+ float c[3];
+} vector_t;
+
+typedef struct points_ {
+ vector_t *v;
+ size_t num_points;
+} points_t;
+
+typedef struct plane_ {
+ vector_t normal;
+ vector_t point;
+} plane_t;
+
+vector_t initVector (float x, float y, float z);
+vector_t normalizeVector(vector_t v);
+
+plane_t initPlane (vector_t normal, vector_t point);
+
+points_t initPoints (const float *x,
+ const float *y, const float *z,
+ size_t num_points);
+void freePoints (points_t *points);
+
+vector_t crossProduct (vector_t v1, vector_t v2);
+float innerProduct (vector_t v1, vector_t v2);
+float distancePointPlane (vector_t point, plane_t plane);
+vector_t addVectors(vector_t v1, vector_t v2);
+vector_t subtractVectors(vector_t v1, vector_t v2);
+vector_t scaleVector(vector_t v1, float f);
+vector_t averagePoints(points_t points);
+vector_t normalOfPoints(points_t points);
+plane_t planeFromPoints(points_t points);
+
+vector_t averageListedPoints(points_t points, list_t list);
+vector_t normalOfListedPoints(points_t points, list_t list);
+plane_t planeFromListedPoints(points_t points, list_t list);
+
+void printPlane(plane_t p);
+void printVector(vector_t v);
diff --git a/src/mtx_qhull/zhull.c b/src/mtx_qhull/zhull.c
new file mode 100644
index 0000000..d13260e
--- /dev/null
+++ b/src/mtx_qhull/zhull.c
@@ -0,0 +1,482 @@
+#define TOL 1e-7
+#define MAXIT 1000000
+
+
+vector_t averageListedPoints(const points_t points, const list_t list) {
+ index_t i;
+ vector_t m = initVector(0.0f, 0.0f, 0.0f);
+ for (i=0; i<getLength(list); i++) {
+ m=addVectors(points.v[getEntry(list,i)],m);
+ }
+ m=scaleVector(m,1.0f/((float)getLength(list)));
+ return m;
+}
+
+vector_t normalOfListedPoints(const points_t points, const list_t list) {
+ vector_t m = averageListedPoints(points,list);
+ vector_t n = initVector(0.0f, 0.0f, 0.0f);
+ vector_t d1,d2,c;
+ index_t i;
+ for (i=1; i<getLength(list); i++) {
+ d1=subtractVectors(points.v[getEntry(list,i-1)],m);
+ d2=subtractVectors(points.v[getEntry(list,i)],m);
+ c=crossProduct(d1,d2);
+ n=addVectors(c,n);
+ }
+ return n;
+}
+
+plane_t planeFromListedPoints(const points_t points, const list_t list) {
+ vector_t p=averageListedPoints(points,list);
+ vector_t n=normalOfListedPoints(points,list);
+ plane_t pl=initPlane(n,p);
+ return pl;
+}
+
+/* facets, memory things */
+
+void freeFacet(facet_t *facet) {
+/* printf("deleting facet %li\n",facet);
+ printList(facet->corners);
+ printList(facet->outsideset);
+ printList(facet->insideset);
+ printList(facet->neighbors);*/
+ freeList(&(facet->corners));
+ freeList(&(facet->outsideset));
+ freeList(&(facet->insideset));
+ freeList(&(facet->neighbors));
+}
+
+list_t appendNewFacets(zhull_t * const zh, const size_t num_facets) {
+ facet_t *new_facet;
+ list_t new_facets=initConstantList(0,num_facets);
+ index_t i;
+ for (i=0; i<getLength(new_facets); i++) {
+ new_facet = (facet_t*) malloc(sizeof(facet_t));
+ if (new_facet==0)
+ break;
+ new_facet->neighbors=emptyList();
+ new_facet->corners=emptyList();
+ new_facet->outsideset=emptyList();
+ new_facet->insideset=emptyList();
+ new_facet->maxdistance=0;
+ new_facet->farthest_outside_point=0;
+// printf("%li, %li\n", new_facet, (entry_t)new_facet);
+ setEntry(new_facets,i,(entry_t)new_facet);
+// printf("created facet %li\n",new_facet);
+ }
+ appendListToList(&(zh->facets),new_facets);
+ return new_facets;
+}
+
+/* facets, interface */
+/*entry_t getFacetCornerByIndex(const facet_t *facet,
+ index_t index_corner) {
+ if (facet!=0)
+ return getEntry(facet->corners,
+ index_corner%getLength(facet->corners));
+ else
+ return 0;
+ }
+ */
+
+facet_t *getFacetByIndex(const list_t facets,
+ const index_t index) {
+ return (facet_t*) getEntry(facets,index);
+}
+
+facet_t *getFacetByPointer(const entry_t ptr) {
+ return (facet_t*) ptr;
+}
+
+void getHorizonEdgeByIndex(index_t *corners,
+ const list_t horizon_fcts, const list_t horizon_fcts_edges,
+ const index_t index) {
+ index_t i=(index+getLength(horizon_fcts_edges))
+ %getLength(horizon_fcts_edges);
+ facet_t *f = getFacetByIndex(horizon_fcts,i);
+ index_t j=getEntry(horizon_fcts_edges,i);
+
+ corners[0]=getEntry(f->corners, j);
+ if (getLength(f->corners)>0)
+ corners[1]=getEntry(f->corners, (j+1) % getLength(f->corners));
+ else
+ corners[1]=corners[0];
+}
+
+
+void removeFacetByPointer(zhull_t * const zh,
+ facet_t * const pointer) {
+ removeValueFromList(&(zh->facets), (entry_t)pointer);
+ removeValueFromList(&(zh->facets_with_outsidepoints),
+ (entry_t)pointer);
+ removeValueFromList(&(zh->facets_with_insidepoints),
+ (entry_t)pointer);
+ freeFacet((facet_t*)pointer);
+}
+
+void removeFacetByPointerList(zhull_t * const zh,
+ const list_t pointers) {
+ index_t i;
+ for (i=0; i<getLength(pointers); i++) {
+ removeFacetByPointer(zh,
+ getFacetByIndex(pointers,i));
+ }
+}
+
+void removeFacetByIndex(zhull_t * const zh,
+ const index_t index) {
+ removeFacetByPointer(zh,
+ getFacetByIndex(zh->facets, index));
+}
+
+void removeFacetByIndexList(zhull_t * const zh,
+ const list_t indices) {
+ facet_t *f;
+ index_t i;
+ for (i=0; i<getLength(indices); i++) {
+ f = getFacetByIndex(zh->facets, getEntry(indices,i));
+ removeFacetByPointer(zh,f);
+ }
+}
+
+
+void freeFacets(zhull_t * const zh) {
+ int i;
+ facet_t *f;
+ if (getLength(zh->facets)>0) {
+ for (i=0; i<getLength(zh->facets); i++) {
+ f=getFacetByIndex(zh->facets,i);
+// printf("deleting facet %li\n",i);
+ freeFacet(f);
+ }
+ freeList(&(zh->facets));
+ }
+}
+
+void freeZhull(zhull_t *zh) {
+ if (zh!=0) {
+// printf("free zhull\n");
+ freeFacets(zh);
+ freeList(&(zh->facets_with_insidepoints));
+ freeList(&(zh->facets_with_outsidepoints));
+ freePoints(&(zh->pts));
+ }
+}
+
+
+// ***********************************
+//
+// interface
+
+
+vector_t getPoint(const zhull_t * const zh,
+ const index_t index) {
+ return (vector_t) zh->pts.v[index];
+}
+
+zhull_t zhullInitPoints(const float *x, const float *y,
+ const float *z, const size_t num_points) {
+ zhull_t zh;
+ zh.pts=initPoints(x,y,z,num_points);
+ zh.facets=emptyList();
+ zh.facets_with_outsidepoints=emptyList();
+ zh.facets_with_insidepoints=emptyList();
+ return zh;
+}
+
+
+
+void dividePointsBetweenNewFacets (
+ zhull_t * const zh, const list_t assoc,
+ const list_t new_facets, const list_t horizon_fcts) {
+ index_t i,j;
+ list_t new_and_horizon_fcts=mergeLists(new_facets,horizon_fcts);
+ printList(new_and_horizon_fcts);
+ uniquefyListEntries(&new_and_horizon_fcts);
+ printList(new_and_horizon_fcts);
+ float d;
+ for (i=0; i<getLength(assoc); i++) {
+ printf("p %d\n", getEntry(assoc,i));
+ for (j=0; j<getLength(new_and_horizon_fcts); j++) {
+ d=distancePointPlane(getPoint(zh,getEntry(assoc,i)),
+ getFacetByIndex(new_and_horizon_fcts,j)->plane);
+ // printf("distance %5.2f to %d\n",d,findValueInList((entry_t)getFacetByIndex(new_and_horizon_fcts,j),zh->facets));
+ /*
+ if ((d>=-TOL) && (d<=TOL)) {
+ appendToList(&(getFacetByIndex(new_facets,j)->insideset),
+ getEntry(assoc,i));
+ if (notInList(getEntry(new_facets,j),zh->facets_with_insidepoints))
+ appendToList(&(zh->facets_with_insidepoints),
+ getEntry(new_facets,j));
+ break;
+ }
+ else */
+ if (d>=-TOL) {
+ appendToList(&(getFacetByIndex(new_and_horizon_fcts,j)->outsideset),
+ getEntry(assoc,i));
+ // printf("appended ");
+ if (notInList(getEntry(new_and_horizon_fcts,j),zh->facets_with_outsidepoints))
+ appendToList(&(zh->facets_with_outsidepoints),getEntry(new_and_horizon_fcts,j));
+ if (getFacetByIndex(new_and_horizon_fcts,j)->maxdistance<d) {
+ getFacetByIndex(new_and_horizon_fcts,j)->maxdistance=d;
+ getFacetByIndex(new_and_horizon_fcts,j)->farthest_outside_point=getEntry(assoc,i);
+ }
+ break;
+ }
+ }
+ }
+}
+
+void zhullInitialFacets(zhull_t *zh) {
+ list_t assoc = emptyList();
+ list_t new_facets = emptyList();
+ index_t i;
+ if (zh->pts.num_points >= 3) {
+ assoc = initListFromTo(0,zh->pts.num_points-1);
+ new_facets = appendNewFacets(zh,2);
+ if (getLength(new_facets)==2) {
+ getFacetByIndex(new_facets,0)->corners = initListFromTo(0,2);
+ getFacetByIndex(new_facets,1)->corners = initListFromTo(2,0);
+ getFacetByIndex(new_facets,0)->neighbors =
+ initConstantList((entry_t)getFacetByIndex(new_facets,1),3);
+ getFacetByIndex(new_facets,1)->neighbors =
+ initConstantList((entry_t)getFacetByIndex(new_facets,0),3);
+ for (i=0; i<2; i++) {
+ getFacetByIndex(new_facets,i)->plane =
+ planeFromListedPoints(zh->pts, getFacetByIndex(new_facets,i)->corners);
+ getFacetByIndex(new_facets,i)->outsideset = emptyList();
+ getFacetByIndex(new_facets,i)->insideset = emptyList();
+ getFacetByIndex(new_facets,i)->maxdistance = 0.0f;
+ removeValueListFromList(&assoc,getFacetByIndex(new_facets,i)->corners);
+ }
+ dividePointsBetweenNewFacets(zh, assoc, new_facets, emptyList());
+ }
+ freeList(&new_facets);
+ freeList(&assoc);
+ }
+}
+
+void printHorizonEdges(list_t *horizon_fcts,
+ list_t *horizon_fcts_edges) {
+ index_t i;
+ index_t c1[2];
+
+ printf("horizon edges: ");
+ for (i=0; i<getLength(*horizon_fcts_edges); i++) {
+ getHorizonEdgeByIndex(c1, *horizon_fcts, *horizon_fcts_edges, i);
+ printf(", %d->%d",c1[0],c1[1]);
+ }
+ printf("\n");
+}
+
+
+void sortHorizonEdges(list_t *horizon_fcts,
+ list_t *horizon_fcts_edges) {
+ index_t i,j;
+ facet_t *fi;
+ index_t ei;
+ index_t c1[2];
+ index_t c2[2];
+
+ if (getLength(*horizon_fcts_edges)==0)
+ return;
+ for (i=0; i<getLength(*horizon_fcts_edges)-1; i++) {
+ getHorizonEdgeByIndex(c1, *horizon_fcts, *horizon_fcts_edges, i);
+ for (j=i+1; j<getLength(*horizon_fcts_edges); j++) {
+ getHorizonEdgeByIndex(c2, *horizon_fcts, *horizon_fcts_edges, j);
+ if (c1[1]==c2[0]) { //found edge continuation: swap positions
+ ei=getEntry(*horizon_fcts,j);
+ setEntry(*horizon_fcts,j,getEntry(*horizon_fcts,i+1));
+ setEntry(*horizon_fcts,i+1,ei);
+ ei=getEntry(*horizon_fcts_edges,j);
+ setEntry(*horizon_fcts_edges,j,getEntry(*horizon_fcts_edges,i+1));
+ setEntry(*horizon_fcts_edges,i+1,ei);
+ break;
+ }
+ // HIER MUSS MEHR LOGIK HINEIN: HORIZONT KANN
+ // FEHLER BEINHALTEN
+ }
+ }
+}
+
+void removeVisibleFacetsGetHorizonAndAvailablePoints(
+ zhull_t * const zh, index_t point_index,
+ facet_t *facet,
+ list_t *horizon_fcts,
+ list_t *horizon_fcts_edges,
+ list_t *avail_points) {
+
+ index_t i,j,k;
+ facet_t *f, *n;
+ float d;
+ list_t visible_fcts = emptyList();
+ list_t indices_for_printing = emptyList();
+
+ *avail_points = emptyList();
+ *horizon_fcts = emptyList();
+ *horizon_fcts_edges = emptyList();
+
+ appendToList(&visible_fcts, (entry_t)facet);
+ for(i=0;i<getLength(visible_fcts);i++) {
+ f=getFacetByIndex(visible_fcts,i);
+ appendListToList(avail_points, f->outsideset);
+ appendListToList(avail_points, f->insideset);
+ for (j=0; j<getLength(f->neighbors); j++) {
+ n=getFacetByIndex(f->neighbors,j);
+ if (notInList((entry_t) n, visible_fcts)) {
+ d=distancePointPlane(getPoint(zh,point_index),
+ n->plane);
+ if (d>-TOL) {
+ if (notInList((entry_t)n,visible_fcts))
+ appendToList(&visible_fcts,(entry_t)n);
+ }
+ else {
+ appendToList(horizon_fcts,(entry_t)n);
+ k=getEntry(f->corners,(j+1)%getLength(f->corners));
+ // printf("searching for corner with %d in\n",k);
+ // printList(n->corners);
+ k=findValueInList(k,n->corners);
+ // printf("found %d\n",k);
+ appendToList(horizon_fcts_edges,k);
+ }
+ }
+ }
+ }
+ printf("removing facets");
+ indices_for_printing=findValueListInList(visible_fcts, zh->facets);
+ printList(indices_for_printing);
+ removeFacetByPointerList(zh,visible_fcts);
+ freeList(&visible_fcts);
+ // printHorizonEdges(horizon_fcts, horizon_fcts_edges);
+ sortHorizonEdges(horizon_fcts, horizon_fcts_edges);
+ printHorizonEdges(horizon_fcts, horizon_fcts_edges);
+}
+
+
+void initNewFacets(zhull_t *zh, int point_index,
+ list_t new_facets, list_t horizon_fcts,
+ list_t horizon_fcts_edges) {
+ index_t i,j;
+ entry_t array[3];
+
+ array[0]=point_index;
+ for (i=0; i<getLength(new_facets); i++) {
+ // printf("adding facet %d\n",getEntry(new_facets,i));
+
+ // corners
+ getHorizonEdgeByIndex(array,horizon_fcts, horizon_fcts_edges,i);
+ array[2]=array[1];
+ array[1]=array[0];
+ array[0]=array[2];
+ array[2]=point_index;
+ getFacetByIndex(new_facets,i)->corners=initList(array,3);
+ // printList(getFacetByIndex(new_facets,i)->corners);
+
+ // neighbors
+ // previous new neighbor
+ j=(getLength(horizon_fcts)+i-1) % getLength(horizon_fcts);
+ array[1]=getEntry(new_facets,j);
+ // next new neighbor
+ j=(i+1) % getLength(horizon_fcts);
+ array[2]=getEntry(new_facets,j);
+ // old neighbor
+ array[0]=getEntry(horizon_fcts,i);
+ getFacetByIndex(new_facets,i)->neighbors=
+ initList(array,3);
+
+ // registering at old neighbor:
+ setEntry(getFacetByIndex(horizon_fcts,i)->neighbors,
+ getEntry(horizon_fcts_edges,i), getEntry(new_facets,i));
+
+ // initializing normal vectors and lists
+ getFacetByIndex(new_facets,i)->plane =
+ planeFromListedPoints(zh->pts,
+ getFacetByIndex(new_facets,i)->corners);
+ }
+}
+
+void makePyramidFacetsToHorizon(zhull_t *zh, index_t point_index,
+ list_t horizon_fcts, list_t horizon_fcts_edges,
+ list_t avail_points) {
+ list_t new_facets = appendNewFacets(zh, getLength(horizon_fcts_edges));
+ initNewFacets(zh,point_index,new_facets,horizon_fcts,horizon_fcts_edges);
+ // printf("available points: ");
+ // printList(avail_points);
+ // printf("new facets : ");
+ // printList(new_facets);
+ dividePointsBetweenNewFacets(zh, avail_points, new_facets, horizon_fcts);
+ freeList(&new_facets);
+}
+
+void calculateZHull(zhull_t *zh,int maxit) {
+ index_t fli=0;
+ index_t pi;
+ facet_t *f;
+ list_t outsideset;
+ int cnt=0;
+ list_t horizon_fcts=emptyList();
+ list_t horizon_fcts_edges=emptyList();
+ list_t available_points=emptyList();
+ if (maxit>MAXIT)
+ maxit=MAXIT;
+ if (zh->pts.num_points!=0){
+ zhullInitialFacets(zh);
+ printZhull(zh);
+ while((getLength(zh->facets_with_outsidepoints)>0)&&(cnt++<maxit)) {
+ printf("//////////////// ITERATION %d ///////\n",cnt);
+ fli%=getLength(zh->facets_with_outsidepoints);
+ f=getFacetByIndex(zh->facets_with_outsidepoints,fli);
+ pi=f->farthest_outside_point;
+ removeVisibleFacetsGetHorizonAndAvailablePoints(zh,pi,f,
+ &horizon_fcts, &horizon_fcts_edges,
+ &available_points);
+ removeValueFromList(&available_points, pi);
+ makePyramidFacetsToHorizon(zh,pi,horizon_fcts,horizon_fcts_edges,
+ available_points);
+ printZhull(zh);
+
+ freeList(&horizon_fcts);
+ freeList(&horizon_fcts_edges);
+ freeList(&available_points);
+ fli++;
+ }
+ }
+}
+
+void printZhull(const zhull_t * const zh) {
+ index_t fi;
+ list_t indices = emptyList();
+ printf("zhull from %d points\n", zh->pts.num_points);
+ printf("facets with outsidepoints: ");
+ indices=findValueListInList(zh->facets_with_outsidepoints,zh->facets);
+ printList(indices);
+ freeList(&indices);
+// printf("facets with insidepoints: ");
+// printList(zh->facets_with_insidepoints);
+ printf("zhull has %d facets\n", getLength(zh->facets));
+ for (fi=0; fi<getLength(zh->facets); fi++) {
+ printf("facet %d<%d>: ",fi,getFacetByIndex(zh->facets,fi));
+ printFacet(zh,getFacetByIndex(zh->facets,fi));
+ }
+}
+
+void printFacet(const zhull_t * const zh,
+ const facet_t * const f) {
+ list_t indices=emptyList();
+ indices=findValueListInList(f->neighbors,zh->facets);
+ printf("plane: ");
+ printPlane(f->plane);
+ printf("\n");
+ printf("corners: ");
+ printList(f->corners);
+ printf("outsideset: ");
+ printList(f->outsideset);
+ printf("insideset: ");
+ printList(f->insideset);
+ printf("neighbors: ");
+ printList(indices);
+ freeList(&indices);
+ printf("pt %d with maxdist %5.2f\n",f->farthest_outside_point, f->maxdistance);
+}
+
diff --git a/src/mtx_qhull/zhull.h b/src/mtx_qhull/zhull.h
new file mode 100644
index 0000000..dd520e1
--- /dev/null
+++ b/src/mtx_qhull/zhull.h
@@ -0,0 +1,25 @@
+
+typedef struct facet_ {
+ plane_t plane;
+ list_t corners;
+ list_t outsideset;
+ list_t insideset;
+ size_t farthest_outside_point;
+ list_t neighbors;
+ float maxdistance;
+} facet_t;
+
+typedef struct zhull_ {
+ points_t pts;
+ list_t facets;
+ list_t facets_with_outsidepoints;
+ list_t facets_with_insidepoints;
+} zhull_t;
+
+void calculateZHull(zhull_t *zh,int maxit);
+void printZhull(const zhull_t * const zh);
+void freeZhull(zhull_t *zh);
+zhull_t zhullInitPoints(const float *x, const float *y,
+ const float *z, const size_t num_points);
+void printFacet(const zhull_t * const zh,
+ const facet_t * const f);