diff options
Diffstat (limited to 'polystat')
-rwxr-xr-x | polystat/help-polystat.pd | 92 | ||||
-rwxr-xr-x | polystat/makefile | 105 | ||||
-rwxr-xr-x | polystat/polystat.c | 268 |
3 files changed, 465 insertions, 0 deletions
diff --git a/polystat/help-polystat.pd b/polystat/help-polystat.pd new file mode 100755 index 0000000..187cf7d --- /dev/null +++ b/polystat/help-polystat.pd @@ -0,0 +1,92 @@ +#N canvas 0 0 785 462 10; +#X obj 76 256 polystat; +#X floatatom 76 351 5 0 0 1 voice - -; +#X floatatom 88 337 5 0 0 1 instances - -; +#X floatatom 101 323 5 0 0 1 voices - -; +#X floatatom 114 309 5 0 0 1 voice - -; +#X obj 127 292 print map; +#X text 195 309 with most instances; +#X obj 76 206 + 7; +#X obj 76 164 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144 -1 +-1; +#X floatatom 76 228 5 0 0 0 - - -; +#X msg 213 184 clear \$1; +#X floatatom 264 145 5 7 38 0 - - -; +#X msg 213 162 8; +#X msg 312 219 voices; +#X msg 312 199 getmap; +#X msg 312 239 clearall; +#X msg 312 159 clear 12; +#X obj 76 184 random 32; +#X obj 26 142 metro 100; +#X obj 26 123 tgl 15 0 empty empty empty 17 7 0 10 -262144 -1 -1 0 +1; +#X obj 247 162 f; +#X obj 247 143 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144 +-1 -1; +#X msg 312 179 print; +#X text 360 199 outputs a list of the instances of voices 0 to HIGHEST +; +#X text 353 180 some stats; +#X msg 340 323 \$1; +#X text 374 239 resets the object; +#X text 374 159 clears a specific voice; +#X obj 26 26 polystat; +#X text 88 26 outputs statistics about voice usage. See also; +#X obj 416 24 polymap; +#X obj 473 24 sieve; +#X text 360 220 outputs a list of all the voices used; +#X text 246 128 clear voice with most instances; +#X text 202 53 most stealing: voice with most instances set to new +value when another voice's instances are equal; +#X text 201 84 first-past-the-post: voice with most instances only +updated when another voice's instances exceed; +#X text 218 144 V; +#X text 218 133 |; +#X text 218 125 |; +#X text 217 113 clear most recent voice; +#X obj 26 79 bng 15 250 50 0 empty empty empty 0 -6 0 8 -262144 -1 +-1; +#X msg 26 97 1; +#X msg 49 97 0; +#X obj 26 59 loadbang; +#X obj 49 121 pipe 2000; +#X msg 151 145 set \$1; +#X msg 150 57 mode 0; +#X msg 150 87 mode 1; +#X obj 515 24 maskxor; +#X text 44 389 Polyphony control objects by Ed Kelly \, November 2006 +; +#X connect 0 0 1 0; +#X connect 0 1 2 0; +#X connect 0 2 3 0; +#X connect 0 3 4 0; +#X connect 0 4 5 0; +#X connect 4 0 25 0; +#X connect 7 0 9 0; +#X connect 7 0 45 0; +#X connect 8 0 17 0; +#X connect 9 0 0 0; +#X connect 10 0 0 0; +#X connect 11 0 20 1; +#X connect 12 0 10 0; +#X connect 13 0 0 0; +#X connect 14 0 0 0; +#X connect 15 0 0 0; +#X connect 16 0 0 0; +#X connect 17 0 7 0; +#X connect 18 0 8 0; +#X connect 19 0 18 0; +#X connect 20 0 10 0; +#X connect 21 0 20 0; +#X connect 22 0 0 0; +#X connect 25 0 11 0; +#X connect 40 0 41 0; +#X connect 40 0 42 0; +#X connect 41 0 19 0; +#X connect 42 0 44 0; +#X connect 43 0 40 0; +#X connect 44 0 19 0; +#X connect 45 0 12 0; +#X connect 46 0 0 0; +#X connect 47 0 0 0; diff --git a/polystat/makefile b/polystat/makefile new file mode 100755 index 0000000..5f30bf3 --- /dev/null +++ b/polystat/makefile @@ -0,0 +1,105 @@ +current: + echo make pd_linux, pd_nt, pd_irix5, pd_irix6 or pd_darwin, then make install + +clean: ; rm -f *.pd_* *.o + +# ----------------------- NT ----------------------- + +pd_nt: polystat.dll + +INSTALL_PREFIX="C:\pd\extra" +EXT=dll +.SUFFIXES: .obj .dll + +PDNTCFLAGS = /W3 /WX /DNT /DPD /nologo +VC="D:\Program Files\Microsoft Visual Studio\Vc98" + +PDNTINCLUDE = /I. /I\tcl\include /I..\..\src /I$(VC)\include + +PDNTLDIR = $(VC)\lib +PDNTLIB = $(PDNTLDIR)\libc.lib \ + $(PDNTLDIR)\oldnames.lib \ + $(PDNTLDIR)\kernel32.lib \ + ..\..\bin\pd.lib + +.c.dll: + cl $(PDNTCFLAGS) $(PDNTINCLUDE) /c $*.c + link /dll /export:$*_setup $*.obj $(PDNTLIB) + +# ----------------------- IRIX 5.x ----------------------- + +pd_irix5: polystat.pd_irix5 + +INSTALL_PREFIX=/usr/local +EXT=pd_irix5 +.SUFFIXES: .pd_irix5 + +SGICFLAGS5 = -o32 -DPD -DUNIX -DIRIX -O2 + +SGIINCLUDE = -I/usr/local/include + +.c.pd_irix5: + cc $(SGICFLAGS5) $(SGIINCLUDE) -o $*.o -c $*.c + ld -elf -shared -rdata_shared -o $*.pd_irix5 $*.o + rm $*.o + +# ----------------------- IRIX 5.x ----------------------- + +pd_irix6: polystat.pd_irix6 + +INSTALL_PREFIX=/usr/local +EXT=pd_irix6 +.SUFFIXES: .pd_irix6 + +SGICFLAGS5 = -o32 -DPD -DUNIX -DIRIX -O2 + +SGIINCLUDE = -I/usr/local/include + +.c.pd_irix6: + cc $(SGICFLAGS5) $(SGIINCLUDE) -o $*.o -c $*.c + ld -elf -shared -rdata_shared -o $*.pd_irix6 $*.o + rm $*.o + +# ----------------------- LINUX i386 ----------------------- + +pd_linux: polystat.pd_linux + +INSTALL_PREFIX=/usr/local +EXT=pd_linux +.SUFFIXES: .pd_linux + +LINUXCFLAGS = -DPD -O2 -funroll-loops -fomit-frame-pointer \ + -Wall -W -Wshadow -Wstrict-prototypes -Werror \ + -Wno-unused -Wno-parentheses -Wno-switch + +LINUXINCLUDE = -I/usr/local/include + +.c.pd_linux: + cc $(LINUXCFLAGS) $(LINUXINCLUDE) -o $*.o -c $*.c + ld -export_dynamic -shared -o $*.pd_linux $*.o -lc -lm + strip --strip-unneeded $*.pd_linux + rm $*.o + +# ----------------------- Mac OSX ----------------------- + +pd_darwin: polystat.pd_darwin + +INSTALL_PREFIX=/usr/local +EXT=pd_darwin +.SUFFIXES: .pd_darwin + +DARWINCFLAGS = -DPD -O2 -Wall -W -Wshadow -Wstrict-prototypes \ + -Wno-unused -Wno-parentheses -Wno-switch + +.c.pd_darwin: + cc $(DARWINCFLAGS) $(LINUXINCLUDE) -o $*.o -c $*.c + cc -bundle -undefined suppress -flat_namespace -o $*.pd_darwin $*.o + rm -f $*.o + +# ---------------------------------------------- + +install:: + install -d $(INSTALL_PREFIX)/lib/pd/extra +# install -m 644 *.$(EXT) $(INSTALL_PREFIX)/lib/pd/externs + install -m 644 polystat.$(EXT) $(INSTALL_PREFIX)/lib/pd/extra + install -m 644 *.pd $(INSTALL_PREFIX)/lib/pd/doc/5.reference diff --git a/polystat/polystat.c b/polystat/polystat.c new file mode 100755 index 0000000..47603fb --- /dev/null +++ b/polystat/polystat.c @@ -0,0 +1,268 @@ +#include "m_pd.h" +#include <math.h> +#include <string.h> +#define MAXENTRIES 512 +#define LASTENTRY 511 + +static t_class *polystat_class; + +typedef struct _map +{ + t_atom map[MAXENTRIES]; + t_atom maxlist[MAXENTRIES]; + t_atom outlist[MAXENTRIES]; +} t_map; + +typedef struct _polystat +{ + t_object x_obj; + t_map x_map; + t_int voices, max, maxval, maxindex, maxflag, maxcount, highest, mode; + t_outlet *voice, *value, *vars, *maxvox, *mapped; +} t_polystat; + +void polystat_float(t_polystat *x, t_floatarg fin) +{ + int voice = (int)fin; + int inst; + int maxindex = (int)x->maxindex + 1; + if(voice >= 0 && voice < MAXENTRIES) + { + int instances = atom_getfloatarg(voice, MAXENTRIES, x->x_map.map); + if(instances == 0)x->voices++; + inst = instances + 1; + SETFLOAT(&x->x_map.map[voice], inst); + if(x->mode == 0) + { + if(inst >= x->max && voice != x->maxval) + { + x->maxflag = x->maxcount >= MAXENTRIES ? 1 : x->maxflag == 1 ? 1 : 0; + x->maxcount += 1; + x->maxcount = x->maxcount > MAXENTRIES ? MAXENTRIES : x->maxcount; + x->maxindex = maxindex % MAXENTRIES; + SETFLOAT(&x->x_map.maxlist[x->maxindex], voice); + x->maxval = voice; + x->max = inst; + outlet_float(x->maxvox, voice); + } + else if(inst >= x->max && voice == x->maxval) + { + x->max = inst; + outlet_float(x->maxvox, voice); + } + } + else if(x->mode != 0) + { + if(inst > x->max && voice != x->maxval) + { + x->maxflag = x->maxcount >= MAXENTRIES ? 1 : x->maxflag == 1 ? 1 : 0; + x->maxcount += 1; + x->maxcount = x->maxcount > MAXENTRIES ? MAXENTRIES : x->maxcount; + x->maxindex = (float)(maxindex % MAXENTRIES); + SETFLOAT(&x->x_map.maxlist[x->maxindex], voice); + x->maxval = voice; + x->max = inst; + outlet_float(x->maxvox, voice); + } + else if(inst > x->max && voice == x->maxval) + { + x->max = inst; + outlet_float(x->maxvox, voice); + } + } + x->highest = voice > x->highest ? voice : x->highest; + outlet_float(x->vars, x->voices); + outlet_float(x->value, inst); + outlet_float(x->voice, voice); + } +} + +void polystat_clear(t_polystat *x, t_floatarg fin) +{ + int inst = (int)fin; + int indexed = (int)(x->maxindex+(MAXENTRIES*x->maxflag)); + int modindex = 0; + int counted = 0; + float current = atom_getfloatarg(inst, MAXENTRIES, x->x_map.map); + float voice; + voice = 0; + if(x->voices > 0 && current > 0) + { + x->voices--; + SETFLOAT(&x->x_map.map[inst], 0); + current = 0; + if(current == 0 && x->maxcount > 0) + { + indexed--; + x->maxcount--; + modindex = (indexed+MAXENTRIES) % MAXENTRIES; + voice = atom_getfloatarg(modindex, MAXENTRIES, x->x_map.maxlist); + current = atom_getfloatarg(voice, MAXENTRIES, x->x_map.map); + } + if(x->maxcount == 0 || current == 0) + { + x->max = 0; + x->maxval = 0; + x->maxindex = 0; + } + else if(x->maxcount > 0 || current > 0) + { + x->max = current; + x->maxval = voice; + x->maxindex = modindex; + } + } + outlet_float(x->maxvox, x->maxval); + outlet_float(x->vars, x->voices); + outlet_float(x->value, x->max); +} + +void polystat_print(t_polystat *x, t_symbol *s) +{ + post("most_instances = %d at voice %d, voices = %d, highest_voice = %d", x->max, x->maxval, x->voices, x->highest); +} + +void polystat_get(t_polystat *x, t_floatarg fin) +{ + int inst = (int)fin; + float voice, instances; + voice = instances = 0; + if(x->voices>0 && inst>=0 && inst<MAXENTRIES) + { + instances = atom_getfloatarg(inst, MAXENTRIES, x->x_map.map); + voice = inst; + } + if(instances>0) + { + outlet_float(x->value, instances); + outlet_float(x->voice, voice); + } +} + +void polystat_clearall(t_polystat *x) +{ + int i; + for(i=0;i<MAXENTRIES;i++) + { + SETFLOAT(&x->x_map.map[i],0); + SETFLOAT(&x->x_map.maxlist[i],0); + } + x->voices = x->max = x->maxval = x->maxindex = x->highest = x->maxcount = 0; + outlet_float(x->vars, x->voices); +} + +void polystat_getmap(t_polystat *x, t_symbol *s) //output map. terminate map at highest voice. +{ + if(x->voices>0) + { + outlet_list(x->mapped, gensym("list"), x->highest+1, x->x_map.map); + } +} + +void polystat_setmap(t_polystat *x, t_symbol *s, int argc, t_atom *argv) +{ + int i; + float arg, max, maxval, high; + max = maxval = high = 0; + for(i=0;i<argc;i++) + { + arg = atom_getfloat(argv+i); + if(arg != 0) + { + if(x->mode==0) + { + if(arg>=maxval) + { + maxval = arg; + max = i; + } + } + else if(x->mode!=0) + { + if(arg>maxval) + { + maxval = arg; + max = i; + } + } + x->highest = i; + SETFLOAT(&x->x_map.map[i], arg); + SETFLOAT(&x->x_map.maxlist[i], 0); + } + } + x->max = max; + x->maxval = maxval; + SETFLOAT(&x->x_map.maxlist[0], max); +} + +void polystat_bang(t_polystat *x, t_symbol *s) +{ + outlet_float(x->vars, x->voices); + outlet_float(x->value, x->maxval); + outlet_float(x->voice, x->max); +} + +void polystat_voices(t_polystat *x, t_symbol *s) //list all voices +{ + int i, mindex, current; + mindex=0; + for(i=0;i<(x->highest+1);i++) + { + current = atom_getfloatarg(i, MAXENTRIES, x->x_map.map); + if(current>0) + { + SETFLOAT(&x->x_map.outlist[mindex],(float)i); + mindex++; + } + } + outlet_list(x->mapped, gensym("list"), x->voices, x->x_map.outlist); +} + +void polystat_mode(t_polystat *x, t_floatarg fmode) +{ + x->mode = fmode; +} + +void *polystat_new(t_floatarg f) +{ + t_polystat *x = (t_polystat *)pd_new(polystat_class); + x->mode = f; + x->max = x->maxval = x->voices = x->maxindex = x->maxcount = x->highest = 0; + //memset(x->x_map.map, 0, MAXENTRIES); + //memset(x->x_map.nomap, 1, MAXENTRIES); + int i; + for(i=0;i<MAXENTRIES;i++) + { + SETFLOAT(x->x_map.map+i, 0); + SETFLOAT(x->x_map.maxlist+i, 0); + } + x->voice = outlet_new(&x->x_obj, &s_float); + x->value = outlet_new(&x->x_obj, &s_float); + x->vars = outlet_new(&x->x_obj, &s_float); + x->maxvox = outlet_new(&x->x_obj, &s_float); + x->mapped = outlet_new(&x->x_obj, &s_list); + return (void *)x; +} + +void polystat_setup(void) +{ + + polystat_class = class_new(gensym("polystat"), + (t_newmethod)polystat_new, + 0, sizeof(t_polystat), + 0, A_DEFFLOAT, 0); + post("|^^^^^^^^^^^^polystat^^^^^^^^^^^^|"); + post("|->^^^^polyphony statistics^^^^<-|"); + post("|^^^^^^^^Edward Kelly 2006^^^^^^^|"); + class_sethelpsymbol(polystat_class, gensym("help-polystat")); + class_addfloat(polystat_class, polystat_float); + class_addmethod(polystat_class, (t_method)polystat_clear, gensym("clear"), A_DEFFLOAT, 0); + class_addmethod(polystat_class, (t_method)polystat_get, gensym("get"), A_DEFFLOAT, 0); + class_addmethod(polystat_class, (t_method)polystat_clearall, gensym("clearall"), A_DEFFLOAT, 0); + class_addmethod(polystat_class, (t_method)polystat_getmap, gensym("getmap"), A_DEFFLOAT, 0); + class_addmethod(polystat_class, (t_method)polystat_setmap, gensym("setmap"), A_GIMME, 0); + class_addmethod(polystat_class, (t_method)polystat_voices, gensym("voices"), A_DEFFLOAT, 0); + class_addmethod(polystat_class, (t_method)polystat_mode, gensym("mode"), A_DEFFLOAT, 0); + class_addmethod(polystat_class, (t_method)polystat_print, gensym("print"), A_DEFFLOAT, 0); + class_addbang(polystat_class, (t_method)polystat_bang); +} |