comparison audio/TimeStretchWrapper.cpp @ 746:771ec060c1d2

Merge from branch audio-source-refactor. Pull out auditioning effect wrapper and time stretch wrapper from play source; corresponding changes to plugin memory management etc.
author Chris Cannam
date Fri, 03 Apr 2020 12:14:05 +0100
parents 846970dbef17
children
comparison
equal deleted inserted replaced
743:7b1d30af4b38 746:771ec060c1d2
1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
2
3 /*
4 Sonic Visualiser
5 An audio file viewer and annotation editor.
6 Centre for Digital Music, Queen Mary, University of London.
7
8 This program is free software; you can redistribute it and/or
9 modify it under the terms of the GNU General Public License as
10 published by the Free Software Foundation; either version 2 of the
11 License, or (at your option) any later version. See the file
12 COPYING included with this distribution for more information.
13 */
14
15 #include "TimeStretchWrapper.h"
16
17 #include <rubberband/RubberBandStretcher.h>
18
19 #include "base/Debug.h"
20
21 using namespace RubberBand;
22 using namespace std;
23
24 TimeStretchWrapper::TimeStretchWrapper(ApplicationPlaybackSource *source) :
25 m_source(source),
26 m_stretcher(nullptr),
27 m_timeRatio(1.0),
28 m_stretcherInputSize(16384),
29 m_channelCount(0),
30 m_sampleRate(0)
31 {
32 }
33
34 TimeStretchWrapper::~TimeStretchWrapper()
35 {
36 delete m_stretcher;
37 }
38
39 void
40 TimeStretchWrapper::setTimeStretchRatio(double ratio)
41 {
42 lock_guard<mutex> guard(m_mutex);
43
44 SVDEBUG << "TimeStretchWrapper::setTimeStretchRatio: setting ratio to "
45 << ratio << " (was " << m_timeRatio << ")" << endl;
46
47 m_timeRatio = ratio;
48
49 // Stretcher will be updated by checkStretcher() from next call to
50 // getSourceSamples()
51 }
52
53 void
54 TimeStretchWrapper::reset()
55 {
56 lock_guard<mutex> guard(m_mutex);
57
58 if (m_stretcher) {
59 m_stretcher->reset();
60 }
61 }
62
63 int
64 TimeStretchWrapper::getSourceSamples(float *const *samples,
65 int nchannels, int nframes)
66 {
67 checkStretcher();
68
69 lock_guard<mutex> guard(m_mutex);
70
71 static int warnings = 0;
72 if (nchannels != m_channelCount) {
73 if (warnings >= 0) {
74 SVCERR << "WARNING: getSourceSamples called for a number of channels different from that set with setSystemPlaybackChannelCount ("
75 << nchannels << " vs " << m_channelCount << ")" << endl;
76 if (++warnings == 6) {
77 SVCERR << "(further warnings will be suppressed)" << endl;
78 warnings = -1;
79 }
80 }
81 return 0;
82 }
83
84 if (!m_stretcher) {
85 return m_source->getSourceSamples(samples, nchannels, nframes);
86 }
87
88 vector<float *> inputPtrs(m_channelCount, nullptr);
89 for (int i = 0; i < m_channelCount; ++i) {
90 inputPtrs[i] = m_inputs[i].data();
91 }
92
93 // The input block for a given output is approx output / ratio,
94 // but we can't predict it exactly, for an adaptive timestretcher.
95
96 sv_frame_t available;
97
98 while ((available = m_stretcher->available()) < nframes) {
99
100 int reqd = int(ceil(double(nframes - available) / m_timeRatio));
101 reqd = std::max(reqd, int(m_stretcher->getSamplesRequired()));
102 reqd = std::min(reqd, m_stretcherInputSize);
103 if (reqd == 0) reqd = 1;
104
105 int got = m_source->getSourceSamples
106 (inputPtrs.data(), nchannels, reqd);
107
108 if (got <= 0) {
109 SVCERR << "WARNING: Failed to obtain any source samples at all"
110 << endl;
111 return 0;
112 }
113
114 m_stretcher->process
115 (inputPtrs.data(), size_t(got), false);
116 }
117
118 return int(m_stretcher->retrieve(samples, nframes));
119 }
120
121 void
122 TimeStretchWrapper::checkStretcher()
123 {
124 lock_guard<mutex> guard(m_mutex);
125
126 if (m_timeRatio == 1.0 || !m_channelCount || !m_sampleRate) {
127 if (m_stretcher) {
128 SVDEBUG << "TimeStretchWrapper::checkStretcher: m_timeRatio = "
129 << m_timeRatio << ", m_channelCount = " << m_channelCount
130 << ", m_sampleRate = " << m_sampleRate
131 << ", deleting existing stretcher" << endl;
132 delete m_stretcher;
133 m_stretcher = nullptr;
134 }
135 return;
136 }
137
138 if (m_stretcher) {
139 SVDEBUG << "TimeStretchWrapper::checkStretcher: setting stretcher ratio to " << m_timeRatio << endl;
140 m_stretcher->setTimeRatio(m_timeRatio);
141 return;
142 }
143
144 SVDEBUG << "TimeStretchWrapper::checkStretcher: creating stretcher with ratio " << m_timeRatio << endl;
145
146 m_stretcher = new RubberBandStretcher
147 (size_t(round(m_sampleRate)),
148 m_channelCount,
149 RubberBandStretcher::OptionProcessRealTime,
150 m_timeRatio);
151
152 m_inputs.resize(m_channelCount);
153 for (auto &v: m_inputs) {
154 v.resize(m_stretcherInputSize);
155 }
156 }
157
158 void
159 TimeStretchWrapper::setSystemPlaybackChannelCount(int count)
160 {
161 {
162 lock_guard<mutex> guard(m_mutex);
163 if (m_channelCount != count) {
164 delete m_stretcher;
165 m_stretcher = nullptr;
166 }
167 m_channelCount = count;
168 }
169 m_source->setSystemPlaybackChannelCount(count);
170 }
171
172 void
173 TimeStretchWrapper::setSystemPlaybackSampleRate(int rate)
174 {
175 {
176 lock_guard<mutex> guard(m_mutex);
177 if (m_sampleRate != rate) {
178 delete m_stretcher;
179 m_stretcher = nullptr;
180 }
181 m_sampleRate = rate;
182 }
183 m_source->setSystemPlaybackSampleRate(rate);
184 }
185
186 std::string
187 TimeStretchWrapper::getClientName() const
188 {
189 return m_source->getClientName();
190 }
191
192 int
193 TimeStretchWrapper::getApplicationSampleRate() const
194 {
195 return m_source->getApplicationSampleRate();
196 }
197
198 int
199 TimeStretchWrapper::getApplicationChannelCount() const
200 {
201 return m_source->getApplicationChannelCount();
202 }
203
204 void
205 TimeStretchWrapper::setSystemPlaybackBlockSize(int sz)
206 {
207 SVDEBUG << "NOTE: TimeStretchWrapper::setSystemPlaybackBlockSize called "
208 << "with size = " << sz << "; not passing to wrapped source, as "
209 << "actual block size will vary" << endl;
210 }
211
212 void
213 TimeStretchWrapper::setSystemPlaybackLatency(int latency)
214 {
215 m_source->setSystemPlaybackLatency(latency);
216 }
217
218 void
219 TimeStretchWrapper::setOutputLevels(float left, float right)
220 {
221 m_source->setOutputLevels(left, right);
222 }
223
224 void
225 TimeStretchWrapper::audioProcessingOverload()
226 {
227 m_source->audioProcessingOverload();
228 }