aboutsummaryrefslogtreecommitdiff
path: root/test/param.h
diff options
context:
space:
mode:
Diffstat (limited to 'test/param.h')
-rw-r--r--test/param.h538
1 files changed, 538 insertions, 0 deletions
diff --git a/test/param.h b/test/param.h
new file mode 100644
index 0000000..4230f8a
--- /dev/null
+++ b/test/param.h
@@ -0,0 +1,538 @@
+
+#include <stdio.h>
+
+struct param {
+ t_symbol* selector; //Type of data stored
+ int ac; //Number of values stored
+ int alloc; //Memory allocated
+ t_atom* av; //Values stored
+ t_symbol* path; //Path(name) of the param
+ t_symbol* path_g;
+ //t_symbol* basepath;
+ struct param* next; //Next param
+ struct param* previous; //Previous param
+ int users; //Number of param objects using this param
+ t_symbol* id; //An id set only if it is saveable
+ int ac_g;
+ t_atom* av_g;
+};
+
+
+struct param* paramlist;
+
+
+typedef struct param_build_info {
+ int ac;
+ t_atom* av;
+ t_symbol* path;
+ t_symbol* id;
+ t_symbol* path_g;
+ int ac_g;
+ t_atom* av_g;
+ t_symbol* basepath;
+ t_symbol* basename;
+}t_param_build_info;
+
+static void set_param_anything( struct param* p, t_symbol* s, int ac, t_atom *av) {
+
+ if ( s == &s_bang ) {
+ p->ac = 0;
+ p->selector = s;
+ } else {
+ if(ac > p->alloc) {
+ p->av = resizebytes(p->av, p->alloc*sizeof(*(p->av)),
+ (10 + ac)*sizeof(*(p->av)));
+ p->alloc = 10 + ac;
+ }
+ p->ac = ac;
+ p->selector = s;
+ tof_copy_atoms(av, p->av, ac);
+ }
+}
+
+
+static void set_param( struct param* p, int ac, t_atom *av) {
+ t_symbol* s;
+ tof_set_selector(&s,&ac, &av );
+ set_param_anything(p,s,ac,av);
+}
+
+
+
+
+
+
+static struct param* get_param_list(void) {
+
+ if (paramlist == NULL) {
+ //post("No params found");
+ return NULL;
+ }
+
+ return paramlist;
+
+}
+
+static void print_all_params(void) {
+
+ struct param* p = paramlist;
+ post("--paramlist--");
+ while(p) {
+ if ( p->path) post("Path: %s",p->path->s_name);
+ if (p->id) post("Id: %s",p->id->s_name);
+ p = p->next;
+ }
+
+}
+
+
+//static struct param* register_param( t_symbol* path, int ac, t_atom* av, t_symbol* id
+static struct param* register_param( t_param_build_info* build) {
+
+ //post("registering %s", path->s_name);
+ t_symbol* path = build->path;
+ t_symbol* path_g = build->path_g;
+ int ac = build->ac;
+ t_atom* av = build->av;
+ t_symbol* id = build->id;
+ int ac_g = build->ac_g;
+ t_atom* av_g = build->av_g;
+
+
+ //if ( path) post("path:%s",path->s_name);
+ //if ( id) post("id:%s",id->s_name);
+
+ struct param* last = paramlist;
+ // Search for param with same path
+ while( last ) {
+ if ( last->path == path) {
+ //post("Found param with same name");
+ last->users = last->users + 1;
+ return last;
+ }
+ if ( last->next == NULL ) break;
+ last = last->next;
+ }
+
+ // Create and add param to the end
+ struct param* p = getbytes(sizeof(*p));
+ p->alloc = 0;
+ p->path = path;
+ p->path_g = path_g;
+ p->next = NULL;
+ p->users = 1;
+ p->id = id;
+ set_param( p, ac, av);
+ p->ac_g = ac_g;
+ p->av_g = getbytes(ac_g*sizeof(*(p->av_g)));
+ tof_copy_atoms(av_g,p->av_g,ac_g);
+ if (last) {
+ //post("Appending param");
+ p->previous = last;
+ last->next = p;
+ } else {
+ //post("Creating first param");
+ p->previous = NULL;
+ paramlist = p;
+ }
+
+
+ //print_all_params();
+
+ return p;
+
+}
+
+static void unregister_param( struct param* p) {
+
+ //post("unregistering %s", p->path->s_name);
+
+ if ( paramlist) {
+
+ p->users = p->users - 1;
+ if ( p->users == 0 ) {
+ // Remove param
+ //post("Removing last param of this name");
+ if (p->previous) {
+ p->previous->next = p->next;
+ if (p->next) p->next->previous = p->previous;
+ /*
+ if (p->next == NULL) {
+ p->previous->next = NULL;
+ } else {
+ p->previous->next = p->next;
+ }
+ */
+ } else {
+ paramlist = p->next;
+ if ( p->next != NULL) p->next->previous = NULL;
+ }
+ freebytes(p->av, p->alloc * sizeof *(p->av) );
+ freebytes(p->av_g, p->ac_g * sizeof *(p->av_g) );
+ freebytes(p, sizeof *p);
+ }
+
+ } else {
+ post("Euh... no params found!");
+ }
+
+ //print_all_params();
+
+}
+
+////////////////////
+
+
+
+static void param_find_value(t_symbol *name, int ac, t_atom *av, int *ac_r,t_atom** av_r) {
+
+ int i;
+ int j = 0;
+ for (i=0;i<ac;i++) {
+ //if ( IS_A_SYMBOL(av,i)) post("analyzing %s",atom_getsymbol(av+i)->s_name);
+ if ( IS_A_SYMBOL(av,i) && name == atom_getsymbol(av+i) && (i+1)<ac ) {
+ //post("matches");
+ i=i+1;
+ for (j=i;j<ac;j++) {
+ if ( IS_A_SYMBOL(av,j) && (atom_getsymbol(av+j))->s_name[0] == '/' ) {
+ //j = j-1;
+ break;
+ }
+ }
+ break;
+ }
+ }
+ j = j-i;
+ //post("i:%d j:%d",i,j);
+
+
+ if ( j > 0) {
+ *ac_r = j;
+ *av_r = av+i;
+ //x->x_param = register_param( x->x_path , j, av+i,saveable);
+ } else {
+ *ac_r = 0;
+ }
+ }
+
+
+static void get_param_build_info(t_canvas* canvas, int o_ac, t_atom* o_av, struct param_build_info* pbi, int flag) {
+
+ pbi->path = NULL;
+ pbi->path_g = NULL;
+ pbi->av = NULL;
+ int saveable = 1;
+ pbi->ac = 0;
+ pbi->id = NULL;
+ pbi->ac_g = 0;
+ pbi->av_g = NULL;
+ pbi->basepath = NULL;
+ pbi->basename = NULL;
+
+ int ac;
+ t_atom* av;
+ tof_get_canvas_arguments(canvas,&ac, &av);
+
+ char *separator = "/";
+ char sbuf_name[MAXPDSTRING];
+ char sbuf_temp[MAXPDSTRING];
+ sbuf_name[0] = '\0';
+ sbuf_temp[0] = '\0';
+
+ t_symbol* id = gensym("/id");
+ t_canvas* id_canvas = canvas;
+ t_canvas* i_canvas = canvas;
+
+ int i;
+
+ int i_ac;
+ t_atom * i_av;
+
+ // A simple flag to indicate if an ID was found
+ int found_id_flag;
+
+
+ t_symbol* id_s = NULL;
+
+ // A HACK to find out if we are in a subpatch or an abstraction
+ // A subpatch always has the same $0 as it's parent
+ //t_symbol* p_id_s = canvas_realizedollar(i_canvas, gensym("$0"));
+ //t_symbol* p_id_s = gensym("");
+
+ while( i_canvas->gl_owner) {
+ // Ignore all supatches
+ if ( tof_canvas_is_not_subpatch(i_canvas) ) {
+ tof_get_canvas_arguments(i_canvas,&i_ac, &i_av);
+ id_s= canvas_realizedollar(i_canvas, gensym("$0"));
+
+ //if (id_s != p_id_s) {
+ // p_id_s = id_s;
+
+ int start = 0;
+ int count = 0;
+ found_id_flag = 0;
+
+ while( tof_get_tagged_argument('/',i_ac,i_av,&start,&count) ) {
+
+ if ( IS_A_SYMBOL(i_av,start)
+ && (id == (i_av+start)->a_w.w_symbol)
+ && (count > 1) ) {
+ id_s = atom_getsymbol(i_av+start+1);
+ id_canvas = i_canvas;
+ found_id_flag = 1;
+ break;
+ }
+ start= start + count;
+ }
+
+
+ // if ever an /id is missing, this param is not saveable
+ if (found_id_flag == 0) saveable = 0;
+
+
+ //if (id_s != p_id_s) {
+ // Prepend newly found ID
+ strcpy(sbuf_temp,sbuf_name);
+ strcpy(sbuf_name, "/");
+ strcat(sbuf_name, id_s->s_name);
+ strcat(sbuf_name,sbuf_temp);
+ //}
+ // p_id_s = id_s;
+
+ //}
+ }
+ i_canvas = i_canvas->gl_owner;
+ }
+
+ if ( saveable ) {
+
+ if ( id_s) {
+
+ int id_buf_length = strlen(id_s->s_name)+2;
+ char * id_buf = getbytes(id_buf_length * sizeof (*id_buf));
+ strcpy(id_buf, "/");
+ strcat(id_buf,id_s->s_name);
+ pbi->id = gensym(id_buf);
+ freebytes(id_buf,id_buf_length * sizeof (*id_buf));
+ } else {
+ pbi->id = gensym("/");
+ }
+
+ }
+
+ pbi->basepath = gensym(sbuf_name);
+
+ if (flag) {
+
+ // FIND NAME
+ t_symbol* midpath = NULL;
+ //t_symbol* basename = NULL;
+
+ if (o_ac && IS_A_SYMBOL(o_av, 0)) {
+ char *firstChar = (atom_getsymbol(o_av))->s_name;
+ if (*firstChar == (char)'/') {
+ strcat(sbuf_name, atom_getsymbol(o_av)->s_name);
+ pbi->path = gensym(sbuf_name);
+ strcat(sbuf_temp, atom_getsymbol(o_av)->s_name);
+ midpath = gensym(sbuf_temp);
+ strcat(sbuf_name, "_");
+ pbi->path_g = gensym(sbuf_name);
+ pbi->basename = atom_getsymbol(o_av);
+ }
+ }
+
+ // if ( x->x_param )
+
+
+ if ( pbi->path) {
+
+
+ // FIND VALUE
+ // A. Find in SUB canvas arguments
+ // B. In canvas' arguments
+ // C. In object's arguments
+ // D. Defaults to a bang
+
+ int p_ac =0;
+ t_atom* p_av;
+
+
+ // A. If name, try to find value in ID canvas' arguments
+ if ( midpath) {
+ // GET ID CANVAS ARGUMENTS (may be the same as the local canvas)
+ //int i_ac;
+ //t_atom * i_av;
+ tof_get_canvas_arguments(id_canvas,&i_ac , &i_av);
+ param_find_value(midpath, i_ac, i_av,&(pbi->ac),&(pbi->av));
+ }
+
+ // B. If basename, try to find value in LOCAL canvas' arguments
+ if ( pbi->basename && pbi->ac == 0 ) {
+
+ param_find_value(pbi->basename, ac, av,&(pbi->ac),&(pbi->av));
+ }
+
+
+ // C. If no value found in canvas' arguments, check the object's arguments
+ if ( pbi->ac == 0 && o_ac > 1) {
+ int start = 1;
+ int count = 0;
+ tof_get_tagged_argument('/',o_ac,o_av,&start,&count);
+ if (count > 0) {
+ pbi->ac = count;
+ pbi->av = o_av + start;
+ }
+ }
+
+ //FIND THE GUI TAGS
+ param_find_value(gensym("/gui"), o_ac, o_av,&(pbi->ac_g),&(pbi->av_g));
+ //post("GUI COUNT:%d",pbi->ac_g);
+
+ }
+
+}
+}
+
+
+static void param_send_prepend(struct param *p, t_symbol* s,t_symbol* prepend) {
+
+ if (p) {
+ if((p->selector == &s_bang)) {
+ // if (s->s_thing)
+ // pd_bang(s->s_thing);
+ } else {
+ if (s->s_thing) {
+ if ( p->selector == &s_list || p->selector == &s_float || p->selector == &s_symbol ) {
+ typedmess(s->s_thing, prepend, p->ac, p->av);
+ } else {
+ int ac = p->ac + 1;
+ t_atom *av = getbytes(ac*sizeof(*av));
+ tof_copy_atoms(p->av,av+1,p->ac);
+ SETSYMBOL(av, p->selector);
+ typedmess(s->s_thing, prepend, ac, av);
+ freebytes(av, ac*sizeof(*av));
+ }
+ }
+ }
+ }
+}
+
+static void param_output(struct param *p, t_outlet* outlet) {
+ // SHOULD I COPY THIS DATA BEFORE SENDING IT OUT?
+ // OR IS THE NORM TO ONLY COPY ON INPUT?
+ if (p) {
+ if((p->selector == &s_bang) ) {
+ outlet_bang(outlet);
+ } else {
+ outlet_anything(outlet, p->selector, p->ac, p->av);
+ }
+ }
+}
+
+
+static void param_output_prepend(struct param* p, t_outlet* outlet, t_symbol* s) {
+
+ if (p->selector == &s_list || p->selector == &s_float || p->selector == &s_symbol) {
+ //t_atom *av = (t_atom *)getbytes(p->ac*sizeof(t_atom));
+ //tof_copy_atoms(p->av,av,p->ac);
+ outlet_anything(outlet,s,p->ac,p->av);
+ //freebytes(av, p->ac*sizeof(t_atom));
+ } else if (p->selector != &s_bang) {
+ int ac = p->ac + 1;
+ t_atom *av = (t_atom *)getbytes(ac*sizeof(t_atom));
+ tof_copy_atoms(p->av,av+1,p->ac);
+ SETSYMBOL(av, p->selector);
+ outlet_anything(outlet,s,ac,av);
+ freebytes(av, ac*sizeof(t_atom));
+ }
+
+}
+
+
+
+static int param_write(t_canvas* canvas, t_symbol* filename, t_symbol* id) {
+
+ int w_error;
+
+ //t_symbol* filename = param_makefilename(basename, n);
+
+ t_binbuf *bbuf = binbuf_new();
+
+ struct param *p = get_param_list();
+ while(p) {
+ if ( p->id && ( id == NULL || p->id == id) && (p->selector != &s_bang)) {
+ int ac = p->ac + 2;
+ t_atom *av = getbytes(ac*sizeof(*av));
+ tof_copy_atoms(p->av,av+2,p->ac);
+ SETSYMBOL(av, p->path);
+ SETSYMBOL(av+1, p->selector);
+ binbuf_add(bbuf, ac, av);
+ binbuf_addsemi(bbuf);
+
+ freebytes(av, ac*sizeof(*av));
+ }
+ p = p->next;
+ }
+
+
+ char buf[MAXPDSTRING];
+ canvas_makefilename(canvas, filename->s_name,
+ buf, MAXPDSTRING);
+
+
+ w_error = (binbuf_write(bbuf, buf, "", 0));
+ //pd_error("%s: write failed", filename->s_name);
+
+ binbuf_free(bbuf);
+
+ return w_error;
+
+}
+
+
+static int param_read(t_canvas* canvas, t_symbol* filename)
+{
+
+ int r_error;
+
+ //t_symbol* filename = param_makefilename(basename, n);
+
+ t_binbuf *bbuf = binbuf_new();
+
+ r_error= (binbuf_read_via_canvas(bbuf, filename->s_name, canvas, 0));
+ //pd_error(x, "%s: read failed", filename->s_name);
+
+
+
+ int bb_ac = binbuf_getnatom(bbuf);
+ int ac = 0;
+ t_atom *bb_av = binbuf_getvec(bbuf);
+ t_atom *av = bb_av;
+
+ while (bb_ac--) {
+ if (bb_av->a_type == A_SEMI) {
+ if ( IS_A_SYMBOL(av,0) && ac > 1) {
+ t_symbol* s = atom_getsymbol(av);
+ if ( s->s_thing) pd_forwardmess(s->s_thing, ac-1, av+1);
+
+
+ }
+
+ ac = 0;
+ av = bb_av + 1;
+ } else {
+
+ ac = ac + 1;
+ }
+ bb_av++;
+ }
+
+
+ binbuf_free(bbuf);
+
+ return r_error;
+}
+
+
+
+