/*
* lifop: a LIFO (last-in first-out) with priorities
*
* (c) 1999-2011 IOhannes m zmölnig, forum::für::umläute, institute of electronic music and acoustics (iem)
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program. If not, see .
*/
#include "zexy.h"
#include
/* ------------------------- lifop ------------------------------- */
/*
* a LIFO (last-in first-out) with priorities
*
* an incoming list is added to a lifo (based on its priority)
* "bang" outputs the last element of the non-empty lifo with the highest priority
*
* high priority means low numeric value
*/
static t_class *lifop_class;
typedef struct _lifop_list {
int argc;
t_atom *argv;
struct _lifop_list *next;
} t_lifop_list;
typedef struct _lifop_prioritylist {
t_float priority;
t_lifop_list *lifo_start;
struct _lifop_prioritylist *next;
} t_lifop_prioritylist;
typedef struct _lifop
{
t_object x_obj;
t_lifop_prioritylist *lifo_list;
unsigned long counter;
t_float priority; /* current priority */
t_outlet *x_out, *x_infout;
} t_lifop;
static t_lifop_prioritylist*lifop_genprioritylist(t_lifop*x, t_float priority)
{
t_lifop_prioritylist*result=0, *dummy=0;
if(x->lifo_list!=0)
{
/*
* do we already have this priority ?
* if so, just return a pointer to that lifo
* else set the dummy-pointer to the lifo BEFORE the new one
*/
dummy=x->lifo_list;
while(dummy!=0){
t_float prio=dummy->priority;
if(prio==priority)return dummy;
if(prio>priority)break;
result=dummy;
dummy=dummy->next;
}
dummy=result;
}
/* create a new priority list */
result = (t_lifop_prioritylist*)getbytes(sizeof( t_lifop_prioritylist));
result->priority=priority;
result->lifo_start=0;
/* insert it into the list of priority lists */
if(dummy==0){
/* insert at the beginning */
result->next=x->lifo_list;
x->lifo_list=result;
} else {
/* post insert into the list of LIFOs */
result->next=dummy->next;
dummy->next =result;
}
/* return the result */
return result;
}
static int add2lifo(t_lifop_prioritylist*lifoprio, int argc, t_atom *argv)
{
t_lifop_list*entry=0;
if(lifoprio==0){
error("plifo: no lifos available");
return -1;
}
/* create an entry for the lifo */
if(!(entry = (t_lifop_list*)getbytes(sizeof(t_lifop_list))))
{
error("plifo: couldn't add entry to end of lifo");
return -1;
}
if(!(entry->argv=(t_atom*)getbytes(argc*sizeof(t_atom)))){
error("plifo: couldn't add list to lifo!");
return -1;
}
memcpy(entry->argv, argv, argc*sizeof(t_atom));
entry->argc=argc;
entry->next=0;
entry->next=lifoprio->lifo_start;
lifoprio->lifo_start=entry;
return 0;
}
static t_lifop_prioritylist*getLifo(t_lifop_prioritylist*plifo)
{
if(plifo==0)return 0;
/* get the highest non-empty lifo */
while(plifo->lifo_start==0 && plifo->next!=0)plifo=plifo->next;
return plifo;
}
static void lifop_list(t_lifop *x, t_symbol *s, int argc, t_atom *argv)
{
t_lifop_prioritylist*plifo=0;
ZEXY_USEVAR(s);
if(!(plifo=lifop_genprioritylist(x, x->priority))) {
error("[lifop]: couldn't get priority lifo");
return;
}
if(!add2lifo(plifo, argc, argv))
{
x->counter++;
}
}
static void lifop_bang(t_lifop *x)
{
t_lifop_prioritylist*plifo=0;
t_lifop_list*lifo=0;
t_atom*argv=0;
int argc=0;
if(!(plifo=getLifo(x->lifo_list))){
outlet_bang(x->x_infout);
return;
}
if(!(lifo=plifo->lifo_start)){
outlet_bang(x->x_infout);
return;
}
x->counter--;
plifo->lifo_start=lifo->next;
/* get the list from the entry */
argc=lifo->argc;
argv=lifo->argv;
lifo->argc=0;
lifo->argv=0;
lifo->next=0;
/* destroy the lifo-entry (important for recursion! */
freebytes(lifo, sizeof(t_lifop_list));
/* output the list */
outlet_list(x->x_out, gensym("list"), argc, argv);
/* free the list */
freebytes(argv, argc*sizeof(t_atom));
}
static void lifop_query(t_lifop*x)
{
z_verbose(1, "%d elements in lifo", (int)x->counter);
outlet_float(x->x_infout, (t_float)x->counter);
}
static void lifop_clear(t_lifop *x)
{
t_lifop_prioritylist *lifo_list=x->lifo_list;
while(lifo_list){
t_lifop_prioritylist *lifo_list2=lifo_list;
t_lifop_list*lifo=lifo_list2->lifo_start;
lifo_list=lifo_list->next;
while(lifo){
t_lifop_list*lifo2=lifo;
lifo=lifo->next;
if(lifo2->argv)freebytes(lifo2->argv, lifo2->argc*sizeof(t_atom));
lifo2->argv=0;
lifo2->argc=0;
lifo2->next=0;
freebytes(lifo2, sizeof(t_lifop_list));
}
lifo_list2->priority =0;
lifo_list2->lifo_start=0;
lifo_list2->next =0;
freebytes(lifo_list2, sizeof( t_lifop_prioritylist));
}
x->lifo_list=0;
x->counter=0;
}
/* this is NOT re-entrant! */
static void lifop_dump(t_lifop*x)
{
t_lifop_prioritylist*plifo=getLifo(x->lifo_list);
if(!plifo||!plifo->lifo_start) {
outlet_bang(x->x_infout);
return;
}
while(plifo) {
t_lifop_list*lifo=plifo->lifo_start;
while(lifo) {
t_atom*argv=lifo->argv;
int argc=lifo->argc;
/* output the list */
outlet_list(x->x_out, gensym("list"), argc, argv);
lifo=lifo->next;
}
plifo=plifo->next;
}
}
static void lifop_free(t_lifop *x)
{
lifop_clear(x);
outlet_free(x->x_out);
outlet_free(x->x_infout);
}
static void *lifop_new(void)
{
t_lifop *x = (t_lifop *)pd_new(lifop_class);
floatinlet_new(&x->x_obj, &x->priority);
x->x_out=outlet_new(&x->x_obj, gensym("list"));
x->x_infout=outlet_new(&x->x_obj, gensym("float"));
x->lifo_list = 0;
x->priority=0;
x->counter=0;
return (x);
}
static void lifop_help(t_lifop*x)
{
post("\n%c lifop\t\t:: a Last-In-First-Out queue with priorities", HEARTSYMBOL);
}
void lifop_setup(void)
{
lifop_class = class_new(gensym("lifop"), (t_newmethod)lifop_new,
(t_method)lifop_free, sizeof(t_lifop), 0, A_NULL);
class_addbang (lifop_class, lifop_bang);
class_addlist (lifop_class, lifop_list);
class_addmethod (lifop_class, (t_method)lifop_clear, gensym("clear"), A_NULL);
class_addmethod (lifop_class, (t_method)lifop_dump, gensym("dump"), A_NULL);
class_addmethod (lifop_class, (t_method)lifop_query, gensym("info"), A_NULL);
class_addmethod (lifop_class, (t_method)lifop_help, gensym("help"), A_NULL);
zexy_register("lifop");
}