diff --git a/pd/src/g_clone.c b/pd/src/g_clone.c index ade8cddd5e9f789e103030608076bee555969e27..89f0fa5b1c0860971faddb781a4216486ec91aba 100644 --- a/pd/src/g_clone.c +++ b/pd/src/g_clone.c @@ -143,6 +143,14 @@ static void clone_in_vis(t_in *x, t_floatarg fn, t_floatarg vis) canvas_vis(x->i_owner->x_vec[n].c_gl, (vis != 0)); } +static void clone_in_fwd(t_in *x, t_symbol *s, int argc, t_atom *argv) +{ + if(s != gensym("fwd")) + typedmess(&x->i_pd, s, argc, argv); + else + pd_error(x->i_owner, "clone-inlet: no method for 'fwd'"); +} + static void clone_out_anything(t_out *x, t_symbol *s, int argc, t_atom *argv) { t_atom *outv; @@ -405,7 +413,8 @@ static void *clone_new(t_symbol *s, int argc, t_atom *argv) obj_issignalinlet(&x->x_vec[0].c_gl->gl_obj, i); x->x_invec[i].i_n = i; if (x->x_invec[i].i_signal) - signalinlet_new(&x->x_obj, 0); + inlet_new(&x->x_obj, &x->x_invec[i].i_pd, + &s_signal, &s_signal); else inlet_new(&x->x_obj, &x->x_invec[i].i_pd, 0, 0); } x->x_nout = obj_noutlets(&x->x_vec[0].c_gl->gl_obj); @@ -458,6 +467,8 @@ void clone_setup(void) A_GIMME, 0); class_addmethod(clone_in_class, (t_method)clone_in_vis, gensym("vis"), A_FLOAT, A_FLOAT, 0); + class_addmethod(clone_in_class, (t_method)clone_in_fwd, gensym("fwd"), + A_GIMME, 0); class_addlist(clone_in_class, (t_method)clone_in_list); clone_out_class = class_new(gensym("clone-outlet"), 0, 0, diff --git a/pd/src/g_io.c b/pd/src/g_io.c index c7330bf635d50c5c9f7c86743b3a0b8047b80653..eee1af66c3906f59210068faa3140745dab6cfa8 100644 --- a/pd/src/g_io.c +++ b/pd/src/g_io.c @@ -36,8 +36,8 @@ typedef struct _vinlet /* if not reblocking, the next slot communicates the parent's inlet signal from the prolog to the DSP routine: */ t_signal *x_directsignal; - - t_resample x_updown; + t_resample x_updown; + t_outlet *x_fwdout; /* optional outlet for forwarding messages to inlet~ */ } t_vinlet; static void *vinlet_new(t_symbol *s) @@ -116,6 +116,19 @@ t_int *vinlet_perform(t_int *w) return (w+4); } +static void vinlet_fwd(t_vinlet *x, t_symbol *s, int argc, t_atom *argv) +{ + if(x->x_fwdout) /* inlet~ fwd */ + outlet_anything(x->x_fwdout, s, argc, argv); + else if(x->x_buf == 0) /* inlet, need to forward message because we want + it to accept fwd selector */ + outlet_anything(x->x_obj.ob_outlet, s, argc, argv); + else /* inlet~ without fwd */ + pd_error(x->x_canvas, "inlet~: expected 'signal' but got '%s' " + "(Note: [inlet~] does not forward non-signal messages unless " + "argument 'fwd' is defined)", s->s_name); +} + static void vinlet_dsp(t_vinlet *x, t_signal **sp) { t_signal *outsig; @@ -251,8 +264,10 @@ static void *vinlet_newsig(t_symbol *s) x->x_endbuf = x->x_buf = (t_float *)getbytes(0); x->x_bufsize = 0; x->x_directsignal = 0; + x->x_fwdout = 0; outlet_new(&x->x_obj, &s_signal); - + /* this line was in pd vanilla but I don't think it is necessary + inlet_new(&x->x_obj, (t_pd *)x->x_inlet, 0, 0); */ resample_init(&x->x_updown); /* this should be thought over: @@ -265,6 +280,8 @@ static void *vinlet_newsig(t_symbol *s) else if (s == gensym("lin"))x->x_updown.method=2; /* up: linear interpolation */ else x->x_updown.method=0; /* up: zero-padding */ + if (s == gensym("fwd")) /* turn on forwarding */ + x->x_fwdout = outlet_new(&x->x_obj, 0); return (x); } @@ -279,6 +296,8 @@ static void vinlet_setup(void) class_addsymbol(vinlet_class, vinlet_symbol); class_addlist(vinlet_class, vinlet_list); class_addanything(vinlet_class, vinlet_anything); + class_addmethod(vinlet_class,(t_method)vinlet_fwd, gensym("fwd"), + A_GIMME, 0); class_addmethod(vinlet_class, (t_method)vinlet_dsp, gensym("dsp"), A_CANT, 0); class_sethelpsymbol(vinlet_class, gensym("pd")); diff --git a/pd/src/m_class.c b/pd/src/m_class.c index a9d296369656927a5b2948915e71858ed620bc23..e0122c39f14de1c0679e1c3b16de121f7e2e6dfe 100644 --- a/pd/src/m_class.c +++ b/pd/src/m_class.c @@ -1077,3 +1077,37 @@ t_gotfn zgetfn(t_pd *x, t_symbol *s) if (m->me_name == s) return(m->me_fun); return(0); } + +t_gotfn zcheckgetfn(t_pd *x, t_symbol *s, t_atomtype arg1, ...) +{ + t_class *c = *x; + t_methodentry *m; + int i, j; + + /* get arg types */ + va_list ap; + int nargs = 0; + t_atomtype args[MAXPDARG+1], curr = arg1; + va_start(ap, arg1); + while (curr != A_NULL && nargs < MAXPDARG) + { + args[nargs++] = curr; + curr = va_arg(ap, t_atomtype); + } + if (curr != A_NULL) error("zcheckgetfn: only 5 arguments are typecheckable"); + args[nargs] = A_NULL; + va_end(ap); + + for (i = c->c_nmethod, m = c->c_methods; i--; m++) + { + if (m->me_name == s) + { + j = 0; + /* both argtype lists are valid, dont need to check whether j < MAXDPARG */ + while(m->me_arg[j] != A_NULL && args[j] != A_NULL + && m->me_arg[j] == args[j]) j++; + if(m->me_arg[j] == A_NULL && args[j] == A_NULL) return(m->me_fun); + } + } + return(0); +} diff --git a/pd/src/m_obj.c b/pd/src/m_obj.c index ea1dd5dd2b73ddcaee60de608b2dbfe9072998e8..725ecf42514a51b4d7274165ecd502d0611070e1 100644 --- a/pd/src/m_obj.c +++ b/pd/src/m_obj.c @@ -80,6 +80,36 @@ static void inlet_wrong(t_inlet *x, t_symbol *s, int argc, t_atom *argv) x->i_symfrom->s_name, s->s_name, type_hint(s, argc, argv, 1)); } + /* forward a message to an inlet~ object */ +static int inlet_fwd(t_inlet *x, t_symbol *s, int argc, t_atom *argv) +{ + if(x->i_symfrom == &s_signal + && zcheckgetfn(x->i_dest, gensym("fwd"), A_GIMME, A_NULL)) + { + mess3(x->i_dest, gensym("fwd"), s, argc, argv); + return 1; + } + return 0; +} + +static int inlet_fwd_symbol(t_inlet *x, t_symbol *s) +{ + t_atom sym; SETSYMBOL(&sym, s); + return inlet_fwd(x, &s_symbol, 1, &sym); +} + +static int inlet_fwd_blob(t_inlet *x, t_blob *st) +{ + t_atom blob; SETBLOB(&blob, st); + return inlet_fwd(x, &s_blob, 1, &blob); +} + +static int inlet_fwd_pointer(t_inlet *x, t_gpointer *gp) +{ + t_atom ptr; SETPOINTER(&ptr, gp); + return inlet_fwd(x, &s_pointer, 1, &ptr); +} + static void inlet_list(t_inlet *x, t_symbol *s, int argc, t_atom *argv); /* LATER figure out how to make these efficient: */ @@ -90,7 +120,8 @@ static void inlet_bang(t_inlet *x) else if (!x->i_symfrom) pd_bang(x->i_dest); else if (x->i_symfrom == &s_list) inlet_list(x, &s_bang, 0, 0); - else inlet_wrong(x, &s_bang, 0, 0); + else if (!inlet_fwd(x, &s_bang, 0, 0)) + inlet_wrong(x, &s_bang, 0, 0); } static void inlet_pointer(t_inlet *x, t_gpointer *gp) @@ -104,7 +135,8 @@ static void inlet_pointer(t_inlet *x, t_gpointer *gp) SETPOINTER(&a, gp); inlet_list(x, &s_pointer, 1, &a); } - else inlet_wrong(x, &s_pointer, 0, 0); + else if (!inlet_fwd_pointer(x, gp)) + inlet_wrong(x, &s_pointer, 0, 0); } static void inlet_float(t_inlet *x, t_float f) @@ -129,7 +161,7 @@ static void inlet_symbol(t_inlet *x, t_symbol *s) if (x->i_symfrom == &s_symbol) pd_vmess(x->i_dest, x->i_symto, "s", s); else if (!x->i_symfrom) pd_symbol(x->i_dest, s); - else + else if (!inlet_fwd_symbol(x, s)) { t_atom a; SETSYMBOL(&a, s); @@ -153,7 +185,7 @@ static void inlet_blob(t_inlet *x, t_blob *st) /* MP20061226 blob type */ /*post("inlet_blob calling pd_blob");*/ pd_blob(x->i_dest, st); } - else + else if (!inlet_fwd_blob(x, st)) { /*post("inlet_blob calling inlet_wrong");*/ inlet_wrong(x, &s_blob, 0, 0); @@ -172,7 +204,8 @@ static void inlet_list(t_inlet *x, t_symbol *s, int argc, t_atom *argv) inlet_float(x, atom_getfloat(argv)); else if (argc==1 && argv->a_type == A_SYMBOL) inlet_symbol(x, atom_getsymbol(argv)); - else inlet_wrong(x, &s_list, 0, 0); + else if (!inlet_fwd(x, &s_list, argc, argv)) + inlet_wrong(x, &s_list, 0, 0); } static void inlet_anything(t_inlet *x, t_symbol *s, int argc, t_atom *argv) @@ -181,7 +214,8 @@ static void inlet_anything(t_inlet *x, t_symbol *s, int argc, t_atom *argv) typedmess(x->i_dest, x->i_symto, argc, argv); else if (!x->i_symfrom) typedmess(x->i_dest, s, argc, argv); - else inlet_wrong(x, s, 0, 0); + else if (!inlet_fwd(x, s, argc, argv)) + inlet_wrong(x, s, 0, 0); } void inlet_free(t_inlet *x)