diff options
-rw-r--r-- | src/liststorage.c | 396 | ||||
-rw-r--r-- | src/z_zexy.c | 1 | ||||
-rw-r--r-- | src/z_zexy.h | 1 |
3 files changed, 398 insertions, 0 deletions
diff --git a/src/liststorage.c b/src/liststorage.c new file mode 100644 index 0000000..e12742d --- /dev/null +++ b/src/liststorage.c @@ -0,0 +1,396 @@ +/****************************************************** + * + * zexy - implementation file + * + * copyleft (c) IOhannes m zmölnig + * + * 1999:forum::für::umläute:2004 + * + * institute of electronic music and acoustics (iem) + * + ****************************************************** + * + * license: GNU General Public License v.2 + * + ******************************************************/ + + +/* + this is heavily based on code from [textfile], + which is part of pd and written by Miller S. Pucket + pd (and thus [textfile]) come with their own license +*/ + +#include "zexy.h" + +#ifdef __WIN32__ +# include <io.h> +#else +# include <unistd.h> +#endif + +#include <stdio.h> +#include <fcntl.h> +#include <string.h> + + + +/* ****************************************************************************** */ +/* liststorage : store several lists in a slots (array of lists of lists) */ + + +/* a list of lists */ +typedef struct _msglist { + int argc; + t_atom *argv; + struct _msglist *next; +} t_msglist; + +typedef struct _liststorage +{ + t_object x_obj; /* everything */ + + t_outlet*x_dataout; /* where the data appears */ + t_outlet*x_infoout; /* where meta-information appears */ + + t_inlet*x_slotin; /* setting the current slot */ + + int x_numslots, x_defaultnumslots; + int x_currentslot; + + t_msglist**x_slots; +} t_liststorage; + +static t_class *liststorage_class; + + +/* ************************************************************************ */ +/* helper functions */ + +static t_msglist*_liststorage_getslot(t_liststorage*x, int slot) { + // post("getting slot %d of %d|%d", slot, 0, x->x_numslots); + if(slot<0 || slot>=x->x_numslots) { pd_error(x, "[liststorage]: attempting to access invalid slot %d", slot); return NULL; } + return x->x_slots[slot]; +} + +static void _liststorage_deletemsglist(t_msglist*list) { + t_msglist*x=list; + while(x) { + t_msglist*y=x; + int i=0; + x=x->next; + + freebytes(y->argv, y->argc*sizeof(t_atom)); + y->argc=0; + y->argv=NULL; + y->next=NULL; + freebytes(y, sizeof(t_msglist)); + } +} + +static void _liststorage_deleteslot(t_liststorage*x, int slot) { + t_msglist*list=_liststorage_getslot(x, slot); + if(list) { + _liststorage_deletemsglist(list); + x->x_slots[slot]=NULL; + } +} + +static t_msglist*_liststorage_newslot(int argc, t_atom*argv) { + t_msglist*slot=getbytes(sizeof(t_msglist)); + int i=0; + + slot->argv=getbytes(sizeof(t_atom)*argc); + for(i=0; i<argc; i++) { + slot->argv[i]=argv[i]; + } + + slot->argc=argc; + slot->next=NULL; + + return slot; +} + + +static t_msglist*_liststorage_add2slot(t_msglist*slot, int argc, t_atom*argv) { + t_msglist*dummy=slot; + t_msglist*newlist=_liststorage_newslot(argc, argv); + if(NULL==slot) { + // post("no data yet: new data is %x", newlist); + return newlist; + } + + while(dummy->next) { + dummy=dummy->next; + } + dummy->next=newlist; + // post("added data to slot @ %x", slot); + return slot; +} + + +static int _liststorage_resize(t_liststorage*x, int size) { + t_msglist**newarray=NULL; + int i=0; + + if(size<0) { + pd_error(x, "[liststorage]: refusing to resize for negative amount of slots"); + return 0; + } + + if(size==x->x_numslots) { + verbose(1, "[liststorate] no need to resize array"); + return size; + } + + /* create a new array */ + newarray=getbytes(sizeof(t_msglist*)*size); + for(i=0; i<size; i++) { + newarray[i]=NULL; + } + + /* copy over all the data from the old array (if there was one */ + i=(size<x->x_numslots)?size:x->x_numslots; + while(i-->0) { + newarray[i]=x->x_slots[i]; + x->x_slots[i]=NULL; + } + + /* delete the old array */ + for(i=0; i<x->x_numslots; i++) { + _liststorage_deleteslot(x, i); + } + freebytes(x->x_slots, sizeof(t_msglist*)); + + /* make the new array the current */ + x->x_slots=newarray; + x->x_numslots=size; + + return size; +} + +static int _liststorage_checkslot(t_liststorage*x, const char*string, const int resize) { + int slot=x->x_currentslot; + t_atom atom; + SETFLOAT(&atom, (t_float)slot); + + if(slot<0) { + if(NULL!=string)pd_error(x, "[liststorage]: %s %d", string, slot); + outlet_anything(x->x_infoout, gensym("invalidslot"), 1, &atom); + return -1; + } + if(slot>=x->x_numslots) { + if(resize) { + _liststorage_resize(x, slot+1); + } else { + if(NULL!=string)pd_error(x, "[liststorage]: %s %d", string, slot); + outlet_anything(x->x_infoout, gensym("invalidslot"), 1, &atom); + return -1; + } + } + return slot; +} +/* ************************************************************************ */ +/* object methods */ + + /* recall all lists from the current slot */ +static void liststorage_bang(t_liststorage *x) +{ + t_atom atom; + t_msglist*list=NULL; + + int slot=_liststorage_checkslot(x, "attempting to read data from invalid slot", 0); + if(slot<0)return; + list=_liststorage_getslot(x, slot); + + while(list) { + outlet_list(x->x_dataout, gensym("list"), list->argc, list->argv); + list=list->next; + } + SETFLOAT(&atom, (t_float)slot); + + /* no need for done: use [t b b b] to get beginning and end of output */ + // outlet_anything(x->x_infoout, gensym("done"), 1, &atom); +} + + /* add a new list to the current slot */ +static void liststorage_add(t_liststorage *x, t_symbol *s, int ac, t_atom *av) +{ + t_msglist*list=NULL; + int slot=_liststorage_checkslot(x, "attempting to add data to invalid slot", 1); + if(slot<0)return; + list=_liststorage_getslot(x, slot); + x->x_slots[slot]=_liststorage_add2slot(x->x_slots[slot], ac, av); +} + + /* clear the current slot */ +static void liststorage_clear(t_liststorage *x) +{ + int slot=_liststorage_checkslot(x, "attempting to clear invalid slot", 0); + if(slot<0)return; + + _liststorage_deleteslot(x, slot); +} + + /* clear all slots */ +static void liststorage_clearall(t_liststorage *x) +{ + int i=0; + for(i=0; i<x->x_numslots; i++) { + _liststorage_deleteslot(x, i); + } +} + + /* insert an empty slot at (before) given position */ +static void liststorage_insert(t_liststorage *x, t_floatarg f) +{ + int current=x->x_currentslot; + int slot=-1; + int i=0; + + x->x_currentslot=f; + slot=_liststorage_checkslot(x, "attempting to insert invalid slot", 1); + x->x_currentslot=current; + + if(slot<0)return; + + _liststorage_resize(x, x->x_numslots+1); + + for(i=x->x_numslots-1; i>slot; i--) { + x->x_slots[i]=x->x_slots[i-1]; + } + x->x_slots[slot]=NULL; +} + + /* get the number of slots */ +static void liststorage_info(t_liststorage *x) +{ + t_atom ap; + SETFLOAT(&ap, (t_float)x->x_numslots); + outlet_anything(x->x_infoout, gensym("numslots"), 1, &ap); +} + + + /* get the number of slots */ +static void liststorage_slot(t_liststorage *x, t_floatarg f) +{ + int slot=f; + x->x_currentslot=slot; + +} + + + /* remove empty slots */ +static void liststorage_compress(t_liststorage *x) +{ + t_msglist**newarray=NULL; + int i=0, j=0; + int size=0; + for(i=0; i<x->x_numslots; i++) { + if(NULL!=x->x_slots[i]) { + + size++; + } + } + + if(size>=x->x_numslots) { + // post("incomressible: %d of %d", size, x->x_numslots); + return; + } + + if(size<x->x_defaultnumslots) + size=x->x_defaultnumslots; + + /* create a new array */ + newarray=getbytes(sizeof(t_msglist*)*size); + for(i=0; i<size; i++) { + newarray[i]=NULL; + } + + /* copy over all the data from the old array (if there was one */ + for(i=0, j=0; i<x->x_numslots; i++) { + if(NULL!=x->x_slots[i]) { + newarray[j]=x->x_slots[i]; + j++; + } + x->x_slots[i]=NULL; + } + + /* delete the old array */ + for(i=0; i<x->x_numslots; i++) { + _liststorage_deleteslot(x, i); + } + freebytes(x->x_slots, sizeof(t_msglist*)); + + /* make the new array the current */ + x->x_slots=newarray; + x->x_numslots=size; +} + + +/* ************************************************************************ */ +/* constructor/destructor */ + +static void liststorage_free(t_liststorage *x) +{ + liststorage_clearall(x); + _liststorage_resize(x, 0); +} + +/* constructor: argument is initial number of slots (can grow) */ +static void *liststorage_new(t_floatarg f) +{ + t_liststorage *x = (t_liststorage *)pd_new(liststorage_class); + int slots=f; + + x->x_slotin=inlet_new(&x->x_obj, &x->x_obj.ob_pd, &s_float, gensym("slot")); + x->x_dataout=outlet_new(&x->x_obj, &s_list); + x->x_infoout=outlet_new(&x->x_obj, 0); + + + + if(slots<=0)slots=20; + x->x_defaultnumslots=slots; + x->x_numslots=0; + x->x_currentslot=0; + x->x_slots=NULL; + + _liststorage_resize(x, x->x_defaultnumslots); + + + + return (x); +} + +void liststorage_setup(void) +{ + liststorage_class = class_new(gensym("liststorage"), (t_newmethod)liststorage_new, + (t_method)liststorage_free, sizeof(t_liststorage), 0, A_DEFFLOAT, 0); + + /* recall all lists from the current slot */ + class_addbang(liststorage_class, (t_method)liststorage_bang); + + /* add a new list to the current slot */ + class_addmethod(liststorage_class, (t_method)liststorage_add, gensym("add"), A_GIMME, 0); + /* clear the current slot */ + class_addmethod(liststorage_class, (t_method)liststorage_clear, gensym("clear"), 0); + /* clear all slots */ + class_addmethod(liststorage_class, (t_method)liststorage_clearall, gensym("clearall"), 0); + + + /* add a new list to the current slot */ + class_addmethod(liststorage_class, (t_method)liststorage_slot, gensym("slot"), A_FLOAT, 0); + + + /* insert an empty slot at (before) given position */ + class_addmethod(liststorage_class, (t_method)liststorage_insert, gensym("insert"), A_DEFFLOAT, 0); + + /* remove empty slots */ + class_addmethod(liststorage_class, (t_method)liststorage_compress, gensym("compress"), 0); + + + /* get the number of slots */ + class_addmethod(liststorage_class, (t_method)liststorage_info, gensym("info"), 0); + + zexy_register("liststorage"); +} diff --git a/src/z_zexy.c b/src/z_zexy.c index 1054a11..7456202 100644 --- a/src/z_zexy.c +++ b/src/z_zexy.c @@ -41,6 +41,7 @@ void z_zexy_setup(void) list2symbol_setup(); /* list2symbol */ lister_setup(); /* lister */ listfind_setup(); /* listfind */ + liststorage_setup(); /* liststorage */ lpt_setup(); /* lpt */ makesymbol_setup(); /* makesymbol */ matchbox_setup(); /* matchbox */ diff --git a/src/z_zexy.h b/src/z_zexy.h index d0ec385..310a168 100644 --- a/src/z_zexy.h +++ b/src/z_zexy.h @@ -39,6 +39,7 @@ void list2lists_setup(void); /* list2lists */ void list2symbol_setup(void); /* list2symbol */ void lister_setup(void); /* lister */ void listfind_setup(void); /* listfind */ +void liststorage_setup(void); /* liststorage */ void lpt_setup(void); /* lpt */ void makesymbol_setup(void); /* makesymbol */ void matchbox_setup(void); /* matchbox */ |