annotate audio/AudioCallbackRecordTarget.cpp @ 610:5ddcbc55b6a2

If audio IO is recreated due to channel count change when recording, make sure it's recreated ready-resumed if we were recording already
author Chris Cannam
date Wed, 08 Aug 2018 15:18:52 +0100
parents 96b605673585
children 37b23e50832b
rev   line source
Chris@476 1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
Chris@476 2
Chris@476 3 /*
Chris@476 4 Sonic Visualiser
Chris@476 5 An audio file viewer and annotation editor.
Chris@476 6 Centre for Digital Music, Queen Mary, University of London.
Chris@476 7
Chris@476 8 This program is free software; you can redistribute it and/or
Chris@476 9 modify it under the terms of the GNU General Public License as
Chris@476 10 published by the Free Software Foundation; either version 2 of the
Chris@476 11 License, or (at your option) any later version. See the file
Chris@476 12 COPYING included with this distribution for more information.
Chris@476 13 */
Chris@476 14
Chris@574 15 #include "AudioCallbackRecordTarget.h"
Chris@476 16
Chris@476 17 #include "base/ViewManagerBase.h"
Chris@476 18 #include "base/TempDirectory.h"
Chris@476 19
Chris@477 20 #include "data/model/WritableWaveFileModel.h"
Chris@476 21
Chris@476 22 #include <QDir>
Chris@575 23 #include <QTimer>
Chris@476 24
Chris@609 25 //#define DEBUG_AUDIO_CALLBACK_RECORD_TARGET 1
Chris@609 26
Chris@609 27 static const int recordUpdateTimeout = 200; // ms
Chris@609 28
Chris@574 29 AudioCallbackRecordTarget::AudioCallbackRecordTarget(ViewManagerBase *manager,
Chris@574 30 QString clientName) :
Chris@476 31 m_viewManager(manager),
Chris@476 32 m_clientName(clientName.toUtf8().data()),
Chris@476 33 m_recording(false),
Chris@476 34 m_recordSampleRate(44100),
Chris@546 35 m_recordChannelCount(2),
Chris@485 36 m_frameCount(0),
Chris@574 37 m_model(0),
Chris@575 38 m_buffers(0),
Chris@575 39 m_bufferCount(0),
Chris@574 40 m_inputLeft(0.f),
Chris@580 41 m_inputRight(0.f),
Chris@580 42 m_levelsSet(false)
Chris@476 43 {
Chris@574 44 m_viewManager->setAudioRecordTarget(this);
Chris@574 45
Chris@574 46 connect(this, SIGNAL(recordStatusChanged(bool)),
Chris@574 47 m_viewManager, SLOT(recordStatusChanged(bool)));
Chris@575 48
Chris@575 49 recreateBuffers();
Chris@476 50 }
Chris@476 51
Chris@574 52 AudioCallbackRecordTarget::~AudioCallbackRecordTarget()
Chris@476 53 {
Chris@574 54 m_viewManager->setAudioRecordTarget(0);
Chris@575 55
Chris@575 56 QMutexLocker locker(&m_bufPtrMutex);
Chris@575 57 for (int c = 0; c < m_bufferCount; ++c) {
Chris@575 58 delete m_buffers[c];
Chris@575 59 }
Chris@575 60 delete[] m_buffers;
Chris@476 61 }
Chris@476 62
Chris@575 63 void
Chris@575 64 AudioCallbackRecordTarget::recreateBuffers()
Chris@575 65 {
Chris@575 66 static int bufferSize = 441000;
Chris@575 67
Chris@575 68 int count = m_recordChannelCount;
Chris@575 69
Chris@575 70 if (count > m_bufferCount) {
Chris@575 71
Chris@575 72 RingBuffer<float> **newBuffers = new RingBuffer<float> *[count];
Chris@575 73 for (int c = 0; c < m_bufferCount; ++c) {
Chris@575 74 newBuffers[c] = m_buffers[c];
Chris@575 75 }
Chris@575 76 for (int c = m_bufferCount; c < count; ++c) {
Chris@575 77 newBuffers[c] = new RingBuffer<float>(bufferSize);
Chris@575 78 }
Chris@575 79
Chris@575 80 // This is the only place where m_buffers is rewritten and
Chris@575 81 // should be the only possible source of contention against
Chris@575 82 // putSamples for this mutex (as the model-updating code is
Chris@575 83 // supposed to run in the same thread as this)
Chris@575 84 QMutexLocker locker(&m_bufPtrMutex);
Chris@575 85 delete[] m_buffers;
Chris@575 86 m_buffers = newBuffers;
Chris@575 87 m_bufferCount = count;
Chris@575 88 }
Chris@575 89 }
Chris@575 90
Chris@559 91 int
Chris@574 92 AudioCallbackRecordTarget::getApplicationSampleRate() const
Chris@559 93 {
Chris@559 94 return 0; // don't care
Chris@559 95 }
Chris@559 96
Chris@559 97 int
Chris@574 98 AudioCallbackRecordTarget::getApplicationChannelCount() const
Chris@559 99 {
Chris@559 100 return m_recordChannelCount;
Chris@559 101 }
Chris@559 102
Chris@476 103 void
Chris@574 104 AudioCallbackRecordTarget::setSystemRecordBlockSize(int)
Chris@476 105 {
Chris@476 106 }
Chris@476 107
Chris@476 108 void
Chris@574 109 AudioCallbackRecordTarget::setSystemRecordSampleRate(int n)
Chris@476 110 {
Chris@476 111 m_recordSampleRate = n;
Chris@476 112 }
Chris@476 113
Chris@476 114 void
Chris@574 115 AudioCallbackRecordTarget::setSystemRecordLatency(int)
Chris@476 116 {
Chris@476 117 }
Chris@476 118
Chris@476 119 void
Chris@574 120 AudioCallbackRecordTarget::setSystemRecordChannelCount(int c)
Chris@546 121 {
Chris@546 122 m_recordChannelCount = c;
Chris@575 123 recreateBuffers();
Chris@546 124 }
Chris@546 125
Chris@546 126 void
Chris@574 127 AudioCallbackRecordTarget::putSamples(const float *const *samples, int, int nframes)
Chris@476 128 {
Chris@575 129 // This may be called from RT context, and in a different thread
Chris@575 130 // from everything else in this class. It takes a mutex that
Chris@575 131 // should almost never be contended (see recreateBuffers())
Chris@575 132 if (!m_recording) return;
Chris@575 133
Chris@575 134 QMutexLocker locker(&m_bufPtrMutex);
Chris@575 135 if (m_buffers && m_bufferCount >= m_recordChannelCount) {
Chris@575 136 for (int c = 0; c < m_recordChannelCount; ++c) {
Chris@575 137 m_buffers[c]->write(samples[c], nframes);
Chris@575 138 }
Chris@575 139 }
Chris@575 140 }
Chris@575 141
Chris@575 142 void
Chris@575 143 AudioCallbackRecordTarget::updateModel()
Chris@575 144 {
Chris@485 145 bool secChanged = false;
Chris@485 146 sv_frame_t frameToEmit = 0;
Chris@485 147
Chris@575 148 int nframes = 0;
Chris@575 149 for (int c = 0; c < m_recordChannelCount; ++c) {
Chris@575 150 if (c == 0 || m_buffers[c]->getReadSpace() < nframes) {
Chris@575 151 nframes = m_buffers[c]->getReadSpace();
Chris@575 152 }
Chris@575 153 }
Chris@485 154
Chris@575 155 if (nframes == 0) {
Chris@609 156 #ifdef DEBUG_AUDIO_CALLBACK_RECORD_TARGET
Chris@609 157 cerr << "AudioCallbackRecordTarget::updateModel: no frames available" << endl;
Chris@609 158 #endif
Chris@609 159 if (m_recording) {
Chris@609 160 QTimer::singleShot(recordUpdateTimeout, this, SLOT(updateModel()));
Chris@609 161 }
Chris@575 162 return;
Chris@575 163 }
Chris@485 164
Chris@575 165 float **samples = new float *[m_recordChannelCount];
Chris@575 166 for (int c = 0; c < m_recordChannelCount; ++c) {
Chris@575 167 samples[c] = new float[nframes];
Chris@575 168 m_buffers[c]->read(samples[c], nframes);
Chris@575 169 }
Chris@485 170
Chris@575 171 m_model->addSamples(samples, nframes);
Chris@485 172
Chris@575 173 for (int c = 0; c < m_recordChannelCount; ++c) {
Chris@575 174 delete[] samples[c];
Chris@575 175 }
Chris@575 176 delete[] samples;
Chris@575 177
Chris@575 178 sv_frame_t priorFrameCount = m_frameCount;
Chris@575 179 m_frameCount += nframes;
Chris@575 180
Chris@575 181 RealTime priorRT =
Chris@575 182 RealTime::frame2RealTime(priorFrameCount, m_recordSampleRate);
Chris@575 183
Chris@575 184 RealTime postRT =
Chris@575 185 RealTime::frame2RealTime(m_frameCount, m_recordSampleRate);
Chris@575 186
Chris@575 187 secChanged = (postRT.sec > priorRT.sec);
Chris@575 188 if (secChanged) {
Chris@575 189 m_model->updateModel();
Chris@575 190 frameToEmit = m_frameCount;
Chris@485 191 }
Chris@485 192
Chris@485 193 if (secChanged) {
Chris@485 194 emit recordDurationChanged(frameToEmit, m_recordSampleRate);
Chris@485 195 }
Chris@575 196
Chris@575 197 if (m_recording) {
Chris@575 198 QTimer::singleShot(1000, this, SLOT(updateModel()));
Chris@575 199 }
Chris@476 200 }
Chris@476 201
Chris@476 202 void
Chris@574 203 AudioCallbackRecordTarget::setInputLevels(float left, float right)
Chris@476 204 {
Chris@574 205 if (left > m_inputLeft) m_inputLeft = left;
Chris@574 206 if (right > m_inputRight) m_inputRight = right;
Chris@580 207 m_levelsSet = true;
Chris@574 208 }
Chris@574 209
Chris@574 210 bool
Chris@574 211 AudioCallbackRecordTarget::getInputLevels(float &left, float &right)
Chris@574 212 {
Chris@574 213 left = m_inputLeft;
Chris@574 214 right = m_inputRight;
Chris@581 215 bool valid = m_levelsSet;
Chris@574 216 m_inputLeft = 0.f;
Chris@574 217 m_inputRight = 0.f;
Chris@581 218 m_levelsSet = false;
Chris@581 219 return valid;
Chris@476 220 }
Chris@476 221
Chris@477 222 void
Chris@574 223 AudioCallbackRecordTarget::modelAboutToBeDeleted()
Chris@477 224 {
Chris@477 225 if (sender() == m_model) {
Chris@477 226 m_model = 0;
Chris@477 227 m_recording = false;
Chris@477 228 }
Chris@477 229 }
Chris@477 230
Chris@483 231 QString
Chris@574 232 AudioCallbackRecordTarget::getRecordContainerFolder()
Chris@508 233 {
Chris@508 234 QDir parent(TempDirectory::getInstance()->getContainingPath());
Chris@508 235 QString subdirname("recorded");
Chris@508 236
Chris@508 237 if (!parent.mkpath(subdirname)) {
Chris@574 238 SVCERR << "ERROR: AudioCallbackRecordTarget::getRecordContainerFolder: Failed to create recorded dir in \"" << parent.canonicalPath() << "\"" << endl;
Chris@508 239 return "";
Chris@508 240 } else {
Chris@508 241 return parent.filePath(subdirname);
Chris@508 242 }
Chris@508 243 }
Chris@508 244
Chris@508 245 QString
Chris@574 246 AudioCallbackRecordTarget::getRecordFolder()
Chris@483 247 {
Chris@508 248 QDir parent(getRecordContainerFolder());
Chris@508 249 QDateTime now = QDateTime::currentDateTime();
Chris@508 250 QString subdirname = QString("%1").arg(now.toString("yyyyMMdd"));
Chris@508 251
Chris@483 252 if (!parent.mkpath(subdirname)) {
Chris@574 253 SVCERR << "ERROR: AudioCallbackRecordTarget::getRecordFolder: Failed to create recorded dir in \"" << parent.canonicalPath() << "\"" << endl;
Chris@483 254 return "";
Chris@483 255 } else {
Chris@483 256 return parent.filePath(subdirname);
Chris@483 257 }
Chris@483 258 }
Chris@483 259
Chris@477 260 WritableWaveFileModel *
Chris@574 261 AudioCallbackRecordTarget::startRecording()
Chris@476 262 {
Chris@575 263 if (m_recording) {
Chris@575 264 SVCERR << "WARNING: AudioCallbackRecordTarget::startRecording: We are already recording" << endl;
Chris@575 265 return 0;
Chris@477 266 }
Chris@477 267
Chris@575 268 m_model = 0;
Chris@575 269 m_frameCount = 0;
Chris@575 270
Chris@575 271 QString folder = getRecordFolder();
Chris@575 272 if (folder == "") return 0;
Chris@575 273 QDir recordedDir(folder);
Chris@575 274
Chris@575 275 QDateTime now = QDateTime::currentDateTime();
Chris@575 276
Chris@575 277 // Don't use QDateTime::toString(Qt::ISODate) as the ":" character
Chris@575 278 // isn't permitted in filenames on Windows
Chris@575 279 QString nowString = now.toString("yyyyMMdd-HHmmss-zzz");
Chris@575 280
Chris@575 281 QString filename = tr("recorded-%1.wav").arg(nowString);
Chris@575 282 QString label = tr("Recorded %1").arg(nowString);
Chris@575 283
Chris@575 284 m_audioFileName = recordedDir.filePath(filename);
Chris@575 285
Chris@575 286 m_model = new WritableWaveFileModel(m_recordSampleRate,
Chris@575 287 m_recordChannelCount,
Chris@575 288 m_audioFileName);
Chris@575 289
Chris@575 290 if (!m_model->isOK()) {
Chris@575 291 SVCERR << "ERROR: AudioCallbackRecordTarget::startRecording: Recording failed"
Chris@575 292 << endl;
Chris@575 293 //!!! and throw?
Chris@575 294 delete m_model;
Chris@575 295 m_model = 0;
Chris@575 296 return 0;
Chris@575 297 }
Chris@575 298
Chris@575 299 m_model->setObjectName(label);
Chris@575 300 m_recording = true;
Chris@575 301
Chris@477 302 emit recordStatusChanged(true);
Chris@575 303
Chris@575 304 QTimer::singleShot(1000, this, SLOT(updateModel()));
Chris@575 305
Chris@477 306 return m_model;
Chris@476 307 }
Chris@476 308
Chris@476 309 void
Chris@574 310 AudioCallbackRecordTarget::stopRecording()
Chris@476 311 {
Chris@575 312 if (!m_recording) {
Chris@575 313 SVCERR << "WARNING: AudioCallbackRecordTarget::startRecording: Not recording" << endl;
Chris@575 314 return;
Chris@477 315 }
Chris@477 316
Chris@575 317 m_recording = false;
Chris@575 318
Chris@575 319 m_bufPtrMutex.lock();
Chris@575 320 m_bufPtrMutex.unlock();
Chris@575 321
Chris@575 322 // buffers should now be up-to-date
Chris@575 323 updateModel();
Chris@575 324
Chris@575 325 m_model->writeComplete();
Chris@575 326 m_model = 0;
Chris@575 327
Chris@477 328 emit recordStatusChanged(false);
Chris@497 329 emit recordCompleted();
Chris@476 330 }
Chris@476 331
Chris@476 332