chris@160
|
1 /**
|
chris@160
|
2 * Copyright (c) 2014, 2015, Enzien Audio Ltd.
|
chris@160
|
3 *
|
chris@160
|
4 * Permission to use, copy, modify, and/or distribute this software for any
|
chris@160
|
5 * purpose with or without fee is hereby granted, provided that the above
|
chris@160
|
6 * copyright notice and this permission notice appear in all copies.
|
chris@160
|
7 *
|
chris@160
|
8 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
|
chris@160
|
9 * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
|
chris@160
|
10 * AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
|
chris@160
|
11 * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
|
chris@160
|
12 * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
|
chris@160
|
13 * OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
chris@160
|
14 * PERFORMANCE OF THIS SOFTWARE.
|
chris@160
|
15 */
|
chris@160
|
16
|
chris@160
|
17 #include "SignalPhasor.h"
|
chris@160
|
18
|
chris@160
|
19 // input phase is in the range of [0,1]. It is independent of o->phase.
|
chris@160
|
20 #if HV_SIMD_AVX
|
chris@160
|
21 static void sPhasor_updatePhase(SignalPhasor *o, float p) {
|
chris@160
|
22 o->phase = _mm256_set_ps(
|
chris@160
|
23 p+1.0f+7.0f*o->step.f2sc, p+1.0f+6.0f*o->step.f2sc,
|
chris@160
|
24 p+1.0f+5.0f*o->step.f2sc, p+1.0f+4.0f*o->step.f2sc,
|
chris@160
|
25 p+1.0f+3.0f*o->step.f2sc, p+1.0f+2.0f*o->step.f2sc,
|
chris@160
|
26 p+1.0f+o->step.f2sc, p+1.0f);
|
chris@160
|
27
|
chris@160
|
28 // ensure that o->phase is still in range [1,2]
|
chris@160
|
29 o->phase = _mm256_or_ps(_mm256_andnot_ps(
|
chris@160
|
30 _mm256_set1_ps(-INFINITY), o->phase), _mm256_set1_ps(1.0f));
|
chris@160
|
31 #elif HV_SIMD_SSE
|
chris@160
|
32 static void sPhasor_updatePhase(SignalPhasor *o, hv_uint32_t p) {
|
chris@160
|
33 o->phase = _mm_set_epi32(3*o->step.s+p, 2*o->step.s+p, o->step.s+p, p);
|
chris@160
|
34 #elif HV_SIMD_NEON
|
chris@160
|
35 static void sPhasor_updatePhase(SignalPhasor *o, hv_uint32_t p) {
|
chris@160
|
36 o->phase = (uint32x4_t) {p, o->step.s+p, 2*o->step.s+p, 3*o->step.s+p};
|
chris@160
|
37 #else // HV_SIMD_NONE
|
chris@160
|
38 static void sPhasor_updatePhase(SignalPhasor *o, hv_uint32_t p) {
|
chris@160
|
39 o->phase = p;
|
chris@160
|
40 #endif
|
chris@160
|
41 }
|
chris@160
|
42
|
chris@160
|
43 static void sPhasor_updateFrequency(SignalPhasor *o, float f, double r) {
|
chris@160
|
44 #if HV_SIMD_AVX
|
chris@160
|
45 o->step.f2sc = (float) (f/r);
|
chris@160
|
46 o->inc = _mm256_set1_ps((float) (8.0f*f/r));
|
chris@160
|
47 sPhasor_updatePhase(o, o->phase[0]);
|
chris@160
|
48 #elif HV_SIMD_SSE
|
chris@160
|
49 o->step.s = (hv_int32_t) (f*(4294967296.0/r));
|
chris@160
|
50 o->inc = _mm_set1_epi32(4*o->step.s);
|
chris@160
|
51 sPhasor_updatePhase(o, (hv_uint32_t) (o->phase[0] & 0xFFFFFFFFL));
|
chris@160
|
52 #elif HV_SIMD_NEON
|
chris@160
|
53 o->step.s = (hv_int32_t) (f*(4294967296.0/r));
|
chris@160
|
54 o->inc = vdupq_n_s32(4*o->step.s);
|
chris@160
|
55 sPhasor_updatePhase(o, vgetq_lane_u32(o->phase, 0));
|
chris@160
|
56 #else // HV_SIMD_NONE
|
chris@160
|
57 o->step.s = (hv_int32_t) (f*(4294967296.0/r));
|
chris@160
|
58 o->inc = o->step.s;
|
chris@160
|
59 // no need to update phase
|
chris@160
|
60 #endif
|
chris@160
|
61 }
|
chris@160
|
62
|
chris@160
|
63 hv_size_t sPhasor_init(SignalPhasor *o, double samplerate) {
|
chris@160
|
64 #if HV_SIMD_AVX
|
chris@160
|
65 o->phase = _mm256_set1_ps(1.0f);
|
chris@160
|
66 o->inc = _mm256_setzero_ps();
|
chris@160
|
67 o->step.f2sc = (float) (1.0/samplerate);
|
chris@160
|
68 #elif HV_SIMD_SSE
|
chris@160
|
69 o->phase = _mm_setzero_si128();
|
chris@160
|
70 o->inc = _mm_setzero_si128();
|
chris@160
|
71 o->step.f2sc = (float) (4294967296.0/samplerate);
|
chris@160
|
72 #elif HV_SIMD_NEON
|
chris@160
|
73 o->phase = vdupq_n_u32(0);
|
chris@160
|
74 o->inc = vdupq_n_s32(0);
|
chris@160
|
75 o->step.f2sc = (float) (4294967296.0/samplerate);
|
chris@160
|
76 #else // HV_SIMD_NONE
|
chris@160
|
77 o->phase = 0;
|
chris@160
|
78 o->inc = 0;
|
chris@160
|
79 o->step.f2sc = (float) (4294967296.0/samplerate);
|
chris@160
|
80 #endif
|
chris@160
|
81 return 0;
|
chris@160
|
82 }
|
chris@160
|
83
|
chris@160
|
84 void sPhasor_onMessage(HvBase *_c, SignalPhasor *o, int letIn, const HvMessage *m) {
|
chris@160
|
85 if (letIn == 1) {
|
chris@160
|
86 if (msg_isFloat(m,0)) {
|
chris@160
|
87 float phase = msg_getFloat(m,0);
|
chris@160
|
88 while (phase < 0.0f) phase += 1.0f; // wrap phase to [0,1]
|
chris@160
|
89 while (phase > 1.0f) phase -= 1.0f;
|
chris@160
|
90 #if HV_SIMD_AVX
|
chris@160
|
91 sPhasor_updatePhase(o, phase);
|
chris@160
|
92 #else // HV_SIMD_SSE || HV_SIMD_NEON || HV_SIMD_NONE
|
chris@160
|
93 sPhasor_updatePhase(o, (hv_int32_t) (phase * 4294967296.0));
|
chris@160
|
94 #endif
|
chris@160
|
95 }
|
chris@160
|
96 }
|
chris@160
|
97 }
|
chris@160
|
98
|
chris@160
|
99 hv_size_t sPhasor_k_init(SignalPhasor *o, float frequency, double samplerate) {
|
chris@160
|
100 sPhasor_updateFrequency(o, frequency, samplerate);
|
chris@160
|
101 sPhasor_updatePhase(o, 0);
|
chris@160
|
102 return 0;
|
chris@160
|
103 }
|
chris@160
|
104
|
chris@160
|
105 void sPhasor_k_onMessage(HvBase *_c, SignalPhasor *o, int letIn, const HvMessage *m) {
|
chris@160
|
106 if (msg_isFloat(m,0)) {
|
chris@160
|
107 switch (letIn) {
|
chris@160
|
108 case 0: sPhasor_updateFrequency(o, msg_getFloat(m,0), ctx_getSampleRate(_c)); break;
|
chris@160
|
109 case 1: {
|
chris@160
|
110 float phase = msg_getFloat(m,0);
|
chris@160
|
111 while (phase < 0.0f) phase += 1.0f; // wrap phase to [0,1]
|
chris@160
|
112 while (phase > 1.0f) phase -= 1.0f;
|
chris@160
|
113 #if HV_SIMD_AVX
|
chris@160
|
114 sPhasor_updatePhase(o, phase);
|
chris@160
|
115 #else // HV_SIMD_SSE || HV_SIMD_NEON || HV_SIMD_NONE
|
chris@160
|
116 sPhasor_updatePhase(o, (hv_uint32_t) (phase * 4294967296.0));
|
chris@160
|
117 #endif
|
chris@160
|
118 break;
|
chris@160
|
119 }
|
chris@160
|
120 default: break;
|
chris@160
|
121 }
|
chris@160
|
122 }
|
chris@160
|
123 }
|