1
/******************************************************
3
* zexy - implementation file
5
* copyleft (c) IOhannes m zm�lnig
7
* 1999:forum::f�r::uml�ute:2004
9
* institute of electronic music and acoustics (iem)
11
******************************************************
13
* license: GNU General Public License v.2
15
******************************************************/
18
finally :: some of the missing binops for signals :: ==~
20
1302:forum::f�r::uml�ute:2000
25
/* ----------------------------- eq_tilde ----------------------------- */
26
static t_class *eq_tilde_class, *scalareq_tilde_class;
28
typedef struct _eq_tilde
34
typedef struct _scalareq_tilde
38
t_float x_g; /* inlet value */
41
static void *eq_tilde_new(t_symbol *s, int argc, t_atom *argv)
44
if (argc > 1) post("==~: extra arguments ignored");
47
t_scalareq_tilde *x = (t_scalareq_tilde *)pd_new(scalareq_tilde_class);
48
floatinlet_new(&x->x_obj, &x->x_g);
49
x->x_g = atom_getfloatarg(0, argc, argv);
50
outlet_new(&x->x_obj, &s_signal);
56
t_eq_tilde *x = (t_eq_tilde *)pd_new(eq_tilde_class);
57
inlet_new(&x->x_obj, &x->x_obj.ob_pd, &s_signal, &s_signal);
58
outlet_new(&x->x_obj, &s_signal);
64
static t_int *eq_tilde_perform(t_int *w)
66
t_sample *in1 = (t_sample *)(w[1]);
67
t_sample *in2 = (t_sample *)(w[2]);
68
t_sample *out = (t_sample *)(w[3]);
70
while (n--) *out++ = (*in1++ == *in2++);
74
static t_int *eq_tilde_perf8(t_int *w)
76
t_sample *in1 = (t_sample *)(w[1]);
77
t_sample *in2 = (t_sample *)(w[2]);
78
t_sample *out = (t_sample *)(w[3]);
80
for (; n; n -= 8, in1 += 8, in2 += 8, out += 8)
82
t_sample f0 = in1[0], f1 = in1[1], f2 = in1[2], f3 = in1[3];
83
t_sample f4 = in1[4], f5 = in1[5], f6 = in1[6], f7 = in1[7];
85
t_sample g0 = in2[0], g1 = in2[1], g2 = in2[2], g3 = in2[3];
86
t_sample g4 = in2[4], g5 = in2[5], g6 = in2[6], g7 = in2[7];
88
out[0] = f0 == g0; out[1] = f1 == g1; out[2] = f2 == g2; out[3] = f3 == g3;
89
out[4] = f4 == g4; out[5] = f5 == g5; out[6] = f6 == g6; out[7] = f7 == g7;
94
static t_int *scalareq_tilde_perform(t_int *w)
96
t_sample *in = (t_sample *)(w[1]);
97
t_sample f = *(t_float *)(w[2]);
98
t_sample *out = (t_sample *)(w[3]);
100
while (n--) *out++ = (*in++ == f);
104
static t_int *scalareq_tilde_perf8(t_int *w)
106
t_sample *in = (t_sample *)(w[1]);
107
t_sample g = *(t_float *)(w[2]);
108
t_sample *out = (t_sample *)(w[3]);
110
for (; n; n -= 8, in += 8, out += 8)
112
t_sample f0 = in[0], f1 = in[1], f2 = in[2], f3 = in[3];
113
t_sample f4 = in[4], f5 = in[5], f6 = in[6], f7 = in[7];
115
out[0] = (f0 == g); out[1] = (f1 == g); out[2] = (f2 == g); out[3] = (f3 == g);
116
out[4] = (f4 == g); out[5] = (f5 == g); out[6] = (f6 == g); out[7] = (f7 == g);
122
static t_int *eq_tilde_performSSE(t_int *w)
124
__m128 *in1 = (__m128 *)(w[1]);
125
__m128 *in2 = (__m128 *)(w[2]);
126
__m128 *out = (__m128 *)(w[3]);
127
int n = (int)(w[4])>>4;
128
const __m128 one = _mm_set1_ps(1.f);
132
xmm0 = _mm_cmpeq_ps(in1[0], in2[0]);
133
out[0] = _mm_and_ps (xmm0 , one);
135
xmm1 = _mm_cmpeq_ps(in1[1], in2[1]);
136
out[1] = _mm_and_ps (xmm1 , one);
138
xmm0 = _mm_cmpeq_ps(in1[2], in2[2]);
139
out[2] = _mm_and_ps (xmm0 , one);
141
xmm1 = _mm_cmpeq_ps(in1[3], in2[3]);
142
out[3] = _mm_and_ps (xmm1 , one);
151
static t_int *scalareq_tilde_performSSE(t_int *w)
153
__m128 *in = (__m128 *)(w[1]);
154
__m128 *out = (__m128 *)(w[3]);
155
float f = *(float *)(w[2]);
156
__m128 scalar = _mm_set1_ps(f);
157
int n = (int)(w[4])>>4;
158
const __m128 one = _mm_set1_ps(1.f);
162
xmm0 = _mm_cmpeq_ps (in[0], scalar);
163
out[0] = _mm_and_ps (xmm0, one);
165
xmm1 = _mm_cmpeq_ps (in[1], scalar);
166
out[1] = _mm_and_ps (xmm1, one);
168
xmm0 = _mm_cmpeq_ps (in[2], scalar);
169
out[2] = _mm_and_ps (xmm0, one);
171
xmm1 = _mm_cmpeq_ps (in[3], scalar);
172
out[3] = _mm_and_ps (xmm1, one);
181
static void eq_tilde_dsp(t_eq_tilde *x, t_signal **sp)
183
t_sample*in1=sp[0]->s_vec;
184
t_sample*in2=sp[1]->s_vec;
185
t_sample*out=sp[2]->s_vec;
193
Z_SIMD_CHKBLOCKSIZE(n)&&
194
Z_SIMD_CHKALIGN(in1)&&
195
Z_SIMD_CHKALIGN(in2)&&
196
Z_SIMD_CHKALIGN(out)&&
197
ZEXY_TYPE_EQUAL(t_sample, float)
200
dsp_add(eq_tilde_performSSE, 4, in1, in2, out, n);
204
dsp_add(eq_tilde_perform, 4, in1, in2, out, n);
206
dsp_add(eq_tilde_perf8, 4, in1, in2, out, n);
209
static void scalareq_tilde_dsp(t_scalareq_tilde *x, t_signal **sp)
211
t_sample*in =sp[0]->s_vec;
212
t_sample*out=sp[1]->s_vec;
217
Z_SIMD_CHKBLOCKSIZE(n)&&
218
Z_SIMD_CHKALIGN(in)&&
219
Z_SIMD_CHKALIGN(out) &&
220
ZEXY_TYPE_EQUAL(t_sample, float)
223
dsp_add(scalareq_tilde_performSSE, 4, in, &x->x_g, out, n);
227
dsp_add(scalareq_tilde_perform, 4, in, &x->x_g, out, n);
229
dsp_add(scalareq_tilde_perf8, 4, in, &x->x_g, out, n);
232
static void eq_tilde_help(t_object*x)
234
post("\n%c &&~\t\t:: test 2 signals for equality", HEARTSYMBOL);
236
void setup_0x3d0x3d0x7e(void)
238
eq_tilde_class = class_new(gensym("==~"), (t_newmethod)eq_tilde_new, 0,
239
sizeof(t_eq_tilde), 0, A_GIMME, 0);
240
class_addmethod(eq_tilde_class, (t_method)eq_tilde_dsp, gensym("dsp"), 0);
241
CLASS_MAINSIGNALIN(eq_tilde_class, t_eq_tilde, x_f);
242
class_addmethod (eq_tilde_class, (t_method)eq_tilde_help, gensym("help"), A_NULL);
243
class_sethelpsymbol(eq_tilde_class, gensym("zigbinops"));
245
scalareq_tilde_class = class_new(gensym("==~"), 0, 0,
246
sizeof(t_scalareq_tilde), 0, 0);
247
CLASS_MAINSIGNALIN(scalareq_tilde_class, t_scalareq_tilde, x_f);
248
class_addmethod(scalareq_tilde_class, (t_method)scalareq_tilde_dsp, gensym("dsp"),
250
class_addmethod (scalareq_tilde_class, (t_method)eq_tilde_help, gensym("help"), A_NULL);
251
class_sethelpsymbol(scalareq_tilde_class, gensym("zigbinops"));
253
zexy_register("==~");