aboutsummaryrefslogtreecommitdiff
path: root/shared/unstable/forky.c
blob: 3383dac2d168ae5cf2e051c17bb2b9d791ebab3b (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
/* Copyright (c) 2003 krzYszcz and others.
 * For information on usage and redistribution, and for a DISCLAIMER OF ALL
 * WARRANTIES, see the file, "LICENSE.txt," in this distribution.  */

/* Put here compilation conditionals supporting older Pd versions. */

#include "m_pd.h"
#include "g_canvas.h"
#include "shared.h"
#include "unstable/forky.h"

#if FORKY_VERSION < 37
/* need this for t_class::c_wb field access */
#include "unstable/pd_imp.h"
#endif

#ifdef KRZYSZCZ
//#define FORKY_DEBUG
#endif

t_pd *forky_newobject(t_symbol *s, int ac, t_atom *av)
{
#if FORKY_VERSION >= 37
    typedmess(&pd_objectmaker, s, ac, av);
    return (pd_newest());
#else
    return (0);
#endif
}

void forky_setsavefn(t_class *c, t_forkysavefn fn)
{
#if FORKY_VERSION >= 37
    class_setsavefn(c, fn);
#else
    if (c->c_wb->w_savefn)
    {
	/* cloning is necessary, because class_setwidget has not been called */
	t_widgetbehavior *wb = getbytes(sizeof(*wb));  /* never freed */
#ifdef FORKY_DEBUG
	fprintf(stderr, "cloning widgetbehavior...\n");
#endif
	*wb = *c->c_wb;
	wb->w_savefn = fn;
	class_setwidget(c, wb);
    }
    else c->c_wb->w_savefn = fn;
#endif
}

void forky_setpropertiesfn(t_class *c, t_forkypropertiesfn fn)
{
#if FORKY_VERSION >= 37
    class_setpropertiesfn(c, fn);
#else
    /* assuming wb has already been either cloned (in forky_setsavefn),
       or defined from scratch -- it is unlikely to ever need props without
       a specialized save (always be sure to set props after save, though). */
    c->c_wb->w_propertiesfn = fn;
#endif
}

/* To be called in a 'dsp' method -- e.g. if there are no feeders, the caller
   might use an optimized version of a 'perform' routine.
   LATER think about replacing 'linetraverser' calls with something faster. */
int forky_hasfeeders(t_object *x, t_glist *glist, int inno, t_symbol *outsym)
{
    t_linetraverser t;
    linetraverser_start(&t, glist);
    while (linetraverser_next(&t))
	if (t.tr_ob2 == x && t.tr_inno == inno
#if FORKY_VERSION >= 36
	    && (!outsym || outsym == outlet_getsymbol(t.tr_outlet))
#endif
	    )
	    return (1);
    return (0);
}

/* Not really a forky, just found no better place to put it in.
   Used in sickle's bitwise signal binops (which use forky_hasfeeders() too).
   Checked against msp2. */
t_int forky_getbitmask(int ac, t_atom *av)
{
    t_int result = 0;
    if (sizeof(shared_t_bitmask) >= sizeof(t_int))
    {
	int nbits = sizeof(t_int) * 8;
	shared_t_bitmask bitmask = 1 << (nbits - 1);
	if (ac > nbits)
	    ac = nbits;
	while (ac--)
	{
	    if (av->a_type == A_FLOAT &&
		(int)av->a_w.w_float)  /* CHECKED */
		result |= bitmask;
	    /* CHECKED symbols are zero */
	    bitmask >>= 1;
	    av++;
	}
	/* CHECKED missing are zero */
#ifdef FORKY_DEBUG
	fprintf(stderr, "mask set to %.8x\n", result);
#endif
    }
    else bug("sizeof(shared_t_bitmask)");
    return (result);
}