Mercurial > hg > svapp
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 } |