| /* |
| * Copyright (c) 2014 The WebRTC project authors. All Rights Reserved. |
| * |
| * Use of this source code is governed by a BSD-style license |
| * that can be found in the LICENSE file in the root of the source |
| * tree. An additional intellectual property rights grant can be found |
| * in the file PATENTS. All contributing project authors may |
| * be found in the AUTHORS file in the root of the source tree. |
| */ |
| |
| #include "modules/audio_device/dummy/file_audio_device.h" |
| |
| #include <string.h> |
| |
| #include "rtc_base/checks.h" |
| #include "rtc_base/logging.h" |
| #include "rtc_base/platform_thread.h" |
| #include "rtc_base/time_utils.h" |
| #include "system_wrappers/include/sleep.h" |
| |
| namespace webrtc { |
| |
| const int kRecordingFixedSampleRate = 48000; |
| const size_t kRecordingNumChannels = 2; |
| const int kPlayoutFixedSampleRate = 48000; |
| const size_t kPlayoutNumChannels = 2; |
| const size_t kPlayoutBufferSize = |
| kPlayoutFixedSampleRate / 100 * kPlayoutNumChannels * 2; |
| const size_t kRecordingBufferSize = |
| kRecordingFixedSampleRate / 100 * kRecordingNumChannels * 2; |
| |
| FileAudioDevice::FileAudioDevice(const char* inputFilename, |
| const char* outputFilename) |
| : _ptrAudioBuffer(NULL), |
| _recordingBuffer(NULL), |
| _playoutBuffer(NULL), |
| _recordingFramesLeft(0), |
| _playoutFramesLeft(0), |
| _recordingBufferSizeIn10MS(0), |
| _recordingFramesIn10MS(0), |
| _playoutFramesIn10MS(0), |
| _playing(false), |
| _recording(false), |
| _lastCallPlayoutMillis(0), |
| _lastCallRecordMillis(0), |
| _outputFilename(outputFilename), |
| _inputFilename(inputFilename) {} |
| |
| FileAudioDevice::~FileAudioDevice() {} |
| |
| int32_t FileAudioDevice::ActiveAudioLayer( |
| AudioDeviceModule::AudioLayer& audioLayer) const { |
| return -1; |
| } |
| |
| AudioDeviceGeneric::InitStatus FileAudioDevice::Init() { |
| return InitStatus::OK; |
| } |
| |
| int32_t FileAudioDevice::Terminate() { |
| return 0; |
| } |
| |
| bool FileAudioDevice::Initialized() const { |
| return true; |
| } |
| |
| int16_t FileAudioDevice::PlayoutDevices() { |
| return 1; |
| } |
| |
| int16_t FileAudioDevice::RecordingDevices() { |
| return 1; |
| } |
| |
| int32_t FileAudioDevice::PlayoutDeviceName(uint16_t index, |
| char name[kAdmMaxDeviceNameSize], |
| char guid[kAdmMaxGuidSize]) { |
| const char* kName = "dummy_device"; |
| const char* kGuid = "dummy_device_unique_id"; |
| if (index < 1) { |
| memset(name, 0, kAdmMaxDeviceNameSize); |
| memset(guid, 0, kAdmMaxGuidSize); |
| memcpy(name, kName, strlen(kName)); |
| memcpy(guid, kGuid, strlen(guid)); |
| return 0; |
| } |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::RecordingDeviceName(uint16_t index, |
| char name[kAdmMaxDeviceNameSize], |
| char guid[kAdmMaxGuidSize]) { |
| const char* kName = "dummy_device"; |
| const char* kGuid = "dummy_device_unique_id"; |
| if (index < 1) { |
| memset(name, 0, kAdmMaxDeviceNameSize); |
| memset(guid, 0, kAdmMaxGuidSize); |
| memcpy(name, kName, strlen(kName)); |
| memcpy(guid, kGuid, strlen(guid)); |
| return 0; |
| } |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::SetPlayoutDevice(uint16_t index) { |
| if (index == 0) { |
| _playout_index = index; |
| return 0; |
| } |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::SetPlayoutDevice( |
| AudioDeviceModule::WindowsDeviceType device) { |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::SetRecordingDevice(uint16_t index) { |
| if (index == 0) { |
| _record_index = index; |
| return _record_index; |
| } |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::SetRecordingDevice( |
| AudioDeviceModule::WindowsDeviceType device) { |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::PlayoutIsAvailable(bool& available) { |
| if (_playout_index == 0) { |
| available = true; |
| return _playout_index; |
| } |
| available = false; |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::InitPlayout() { |
| MutexLock lock(&mutex_); |
| |
| if (_playing) { |
| return -1; |
| } |
| |
| _playoutFramesIn10MS = static_cast<size_t>(kPlayoutFixedSampleRate / 100); |
| |
| if (_ptrAudioBuffer) { |
| // Update webrtc audio buffer with the selected parameters |
| _ptrAudioBuffer->SetPlayoutSampleRate(kPlayoutFixedSampleRate); |
| _ptrAudioBuffer->SetPlayoutChannels(kPlayoutNumChannels); |
| } |
| return 0; |
| } |
| |
| bool FileAudioDevice::PlayoutIsInitialized() const { |
| return _playoutFramesIn10MS != 0; |
| } |
| |
| int32_t FileAudioDevice::RecordingIsAvailable(bool& available) { |
| if (_record_index == 0) { |
| available = true; |
| return _record_index; |
| } |
| available = false; |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::InitRecording() { |
| MutexLock lock(&mutex_); |
| |
| if (_recording) { |
| return -1; |
| } |
| |
| _recordingFramesIn10MS = static_cast<size_t>(kRecordingFixedSampleRate / 100); |
| |
| if (_ptrAudioBuffer) { |
| _ptrAudioBuffer->SetRecordingSampleRate(kRecordingFixedSampleRate); |
| _ptrAudioBuffer->SetRecordingChannels(kRecordingNumChannels); |
| } |
| return 0; |
| } |
| |
| bool FileAudioDevice::RecordingIsInitialized() const { |
| return _recordingFramesIn10MS != 0; |
| } |
| |
| int32_t FileAudioDevice::StartPlayout() { |
| if (_playing) { |
| return 0; |
| } |
| |
| _playing = true; |
| _playoutFramesLeft = 0; |
| |
| if (!_playoutBuffer) { |
| _playoutBuffer = new int8_t[kPlayoutBufferSize]; |
| } |
| if (!_playoutBuffer) { |
| _playing = false; |
| return -1; |
| } |
| |
| // PLAYOUT |
| if (!_outputFilename.empty()) { |
| _outputFile = FileWrapper::OpenWriteOnly(_outputFilename.c_str()); |
| if (!_outputFile.is_open()) { |
| RTC_LOG(LS_ERROR) << "Failed to open playout file: " << _outputFilename; |
| _playing = false; |
| delete[] _playoutBuffer; |
| _playoutBuffer = NULL; |
| return -1; |
| } |
| } |
| |
| _ptrThreadPlay = rtc::PlatformThread::SpawnJoinable( |
| [this] { |
| while (PlayThreadProcess()) { |
| } |
| }, |
| "webrtc_audio_module_play_thread", |
| rtc::ThreadAttributes().SetPriority(rtc::ThreadPriority::kRealtime)); |
| |
| RTC_LOG(LS_INFO) << "Started playout capture to output file: " |
| << _outputFilename; |
| return 0; |
| } |
| |
| int32_t FileAudioDevice::StopPlayout() { |
| { |
| MutexLock lock(&mutex_); |
| _playing = false; |
| } |
| |
| // stop playout thread first |
| if (!_ptrThreadPlay.empty()) |
| _ptrThreadPlay.Finalize(); |
| |
| MutexLock lock(&mutex_); |
| |
| _playoutFramesLeft = 0; |
| delete[] _playoutBuffer; |
| _playoutBuffer = NULL; |
| _outputFile.Close(); |
| |
| RTC_LOG(LS_INFO) << "Stopped playout capture to output file: " |
| << _outputFilename; |
| return 0; |
| } |
| |
| bool FileAudioDevice::Playing() const { |
| return _playing; |
| } |
| |
| int32_t FileAudioDevice::StartRecording() { |
| _recording = true; |
| |
| // Make sure we only create the buffer once. |
| _recordingBufferSizeIn10MS = |
| _recordingFramesIn10MS * kRecordingNumChannels * 2; |
| if (!_recordingBuffer) { |
| _recordingBuffer = new int8_t[_recordingBufferSizeIn10MS]; |
| } |
| |
| if (!_inputFilename.empty()) { |
| _inputFile = FileWrapper::OpenReadOnly(_inputFilename.c_str()); |
| if (!_inputFile.is_open()) { |
| RTC_LOG(LS_ERROR) << "Failed to open audio input file: " |
| << _inputFilename; |
| _recording = false; |
| delete[] _recordingBuffer; |
| _recordingBuffer = NULL; |
| return -1; |
| } |
| } |
| |
| _ptrThreadRec = rtc::PlatformThread::SpawnJoinable( |
| [this] { |
| while (RecThreadProcess()) { |
| } |
| }, |
| "webrtc_audio_module_capture_thread", |
| rtc::ThreadAttributes().SetPriority(rtc::ThreadPriority::kRealtime)); |
| |
| RTC_LOG(LS_INFO) << "Started recording from input file: " << _inputFilename; |
| |
| return 0; |
| } |
| |
| int32_t FileAudioDevice::StopRecording() { |
| { |
| MutexLock lock(&mutex_); |
| _recording = false; |
| } |
| |
| if (!_ptrThreadRec.empty()) |
| _ptrThreadRec.Finalize(); |
| |
| MutexLock lock(&mutex_); |
| _recordingFramesLeft = 0; |
| if (_recordingBuffer) { |
| delete[] _recordingBuffer; |
| _recordingBuffer = NULL; |
| } |
| _inputFile.Close(); |
| |
| RTC_LOG(LS_INFO) << "Stopped recording from input file: " << _inputFilename; |
| return 0; |
| } |
| |
| bool FileAudioDevice::Recording() const { |
| return _recording; |
| } |
| |
| int32_t FileAudioDevice::InitSpeaker() { |
| return -1; |
| } |
| |
| bool FileAudioDevice::SpeakerIsInitialized() const { |
| return false; |
| } |
| |
| int32_t FileAudioDevice::InitMicrophone() { |
| return 0; |
| } |
| |
| bool FileAudioDevice::MicrophoneIsInitialized() const { |
| return true; |
| } |
| |
| int32_t FileAudioDevice::SpeakerVolumeIsAvailable(bool& available) { |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::SetSpeakerVolume(uint32_t volume) { |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::SpeakerVolume(uint32_t& volume) const { |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::MaxSpeakerVolume(uint32_t& maxVolume) const { |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::MinSpeakerVolume(uint32_t& minVolume) const { |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::MicrophoneVolumeIsAvailable(bool& available) { |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::SetMicrophoneVolume(uint32_t volume) { |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::MicrophoneVolume(uint32_t& volume) const { |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::MaxMicrophoneVolume(uint32_t& maxVolume) const { |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::MinMicrophoneVolume(uint32_t& minVolume) const { |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::SpeakerMuteIsAvailable(bool& available) { |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::SetSpeakerMute(bool enable) { |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::SpeakerMute(bool& enabled) const { |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::MicrophoneMuteIsAvailable(bool& available) { |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::SetMicrophoneMute(bool enable) { |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::MicrophoneMute(bool& enabled) const { |
| return -1; |
| } |
| |
| int32_t FileAudioDevice::StereoPlayoutIsAvailable(bool& available) { |
| available = true; |
| return 0; |
| } |
| int32_t FileAudioDevice::SetStereoPlayout(bool enable) { |
| return 0; |
| } |
| |
| int32_t FileAudioDevice::StereoPlayout(bool& enabled) const { |
| enabled = true; |
| return 0; |
| } |
| |
| int32_t FileAudioDevice::StereoRecordingIsAvailable(bool& available) { |
| available = true; |
| return 0; |
| } |
| |
| int32_t FileAudioDevice::SetStereoRecording(bool enable) { |
| return 0; |
| } |
| |
| int32_t FileAudioDevice::StereoRecording(bool& enabled) const { |
| enabled = true; |
| return 0; |
| } |
| |
| int32_t FileAudioDevice::PlayoutDelay(uint16_t& delayMS) const { |
| return 0; |
| } |
| |
| void FileAudioDevice::AttachAudioBuffer(AudioDeviceBuffer* audioBuffer) { |
| MutexLock lock(&mutex_); |
| |
| _ptrAudioBuffer = audioBuffer; |
| |
| // Inform the AudioBuffer about default settings for this implementation. |
| // Set all values to zero here since the actual settings will be done by |
| // InitPlayout and InitRecording later. |
| _ptrAudioBuffer->SetRecordingSampleRate(0); |
| _ptrAudioBuffer->SetPlayoutSampleRate(0); |
| _ptrAudioBuffer->SetRecordingChannels(0); |
| _ptrAudioBuffer->SetPlayoutChannels(0); |
| } |
| |
| bool FileAudioDevice::PlayThreadProcess() { |
| if (!_playing) { |
| return false; |
| } |
| int64_t currentTime = rtc::TimeMillis(); |
| mutex_.Lock(); |
| |
| if (_lastCallPlayoutMillis == 0 || |
| currentTime - _lastCallPlayoutMillis >= 10) { |
| mutex_.Unlock(); |
| _ptrAudioBuffer->RequestPlayoutData(_playoutFramesIn10MS); |
| mutex_.Lock(); |
| |
| _playoutFramesLeft = _ptrAudioBuffer->GetPlayoutData(_playoutBuffer); |
| RTC_DCHECK_EQ(_playoutFramesIn10MS, _playoutFramesLeft); |
| if (_outputFile.is_open()) { |
| _outputFile.Write(_playoutBuffer, kPlayoutBufferSize); |
| } |
| _lastCallPlayoutMillis = currentTime; |
| } |
| _playoutFramesLeft = 0; |
| mutex_.Unlock(); |
| |
| int64_t deltaTimeMillis = rtc::TimeMillis() - currentTime; |
| if (deltaTimeMillis < 10) { |
| SleepMs(10 - deltaTimeMillis); |
| } |
| |
| return true; |
| } |
| |
| bool FileAudioDevice::RecThreadProcess() { |
| if (!_recording) { |
| return false; |
| } |
| |
| int64_t currentTime = rtc::TimeMillis(); |
| mutex_.Lock(); |
| |
| if (_lastCallRecordMillis == 0 || currentTime - _lastCallRecordMillis >= 10) { |
| if (_inputFile.is_open()) { |
| if (_inputFile.Read(_recordingBuffer, kRecordingBufferSize) > 0) { |
| _ptrAudioBuffer->SetRecordedBuffer(_recordingBuffer, |
| _recordingFramesIn10MS); |
| } else { |
| _inputFile.Rewind(); |
| } |
| _lastCallRecordMillis = currentTime; |
| mutex_.Unlock(); |
| _ptrAudioBuffer->DeliverRecordedData(); |
| mutex_.Lock(); |
| } |
| } |
| |
| mutex_.Unlock(); |
| |
| int64_t deltaTimeMillis = rtc::TimeMillis() - currentTime; |
| if (deltaTimeMillis < 10) { |
| SleepMs(10 - deltaTimeMillis); |
| } |
| |
| return true; |
| } |
| |
| } // namespace webrtc |