/**************************************************************************** Copyright (c) 2014-2016 Chukong Technologies Inc. Copyright (c) 2017-2022 Xiamen Yaji Software Co., Ltd. http://www.cocos.com Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated engine source code (the "Software"), a limited, worldwide, royalty-free, non-assignable, revocable and non-exclusive license to use Cocos Creator solely to develop games on your target platforms. You shall not use Cocos Creator software for developing other software or tools that's used for developing games. You are not granted to publish, distribute, sublicense, and/or sell copies of Cocos Creator. The software or tools in this License Agreement are licensed, not sold. Xiamen Yaji Software Co., Ltd. reserves all rights not expressly granted to you. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. ****************************************************************************/ #define LOG_TAG "AudioPlayer" #import #include "audio/apple/AudioCache.h" #include "audio/apple/AudioDecoder.h" #include "audio/apple/AudioPlayer.h" #include "base/memory/Memory.h" #include "platform/FileUtils.h" #ifdef VERY_VERY_VERBOSE_LOGGING #define ALOGVV ALOGV #else #define ALOGVV(...) \ do { \ } while (false) #endif using namespace cc; namespace { unsigned int __idIndex = 0; } AudioPlayer::AudioPlayer() : _audioCache(nullptr), _finishCallbak(nullptr), _isDestroyed(false), _removeByAudioEngine(false), _ready(false), _currTime(0.0f), _streamingSource(false), _rotateBufferThread(nullptr), _timeDirty(false), _isRotateThreadExited(false), _needWakeupRotateThread(false), _id(++__idIndex) { memset(_bufferIds, 0, sizeof(_bufferIds)); } AudioPlayer::~AudioPlayer() { ALOGVV("~AudioPlayer() (%p), id=%u", this, _id); destroy(); if (_streamingSource) { alDeleteBuffers(QUEUEBUFFER_NUM, _bufferIds); } } void AudioPlayer::destroy() { if (_isDestroyed) return; ALOGVV("AudioPlayer::destroy begin, id=%u", _id); _isDestroyed = true; do { if (_audioCache != nullptr) { if (_audioCache->_state == AudioCache::State::INITIAL) { ALOGV("AudioPlayer::destroy, id=%u, cache isn't ready!", _id); break; } while (!_audioCache->_isLoadingFinished) { std::this_thread::sleep_for(std::chrono::milliseconds(5)); } } // Wait for play2d to be finished. _play2dMutex.lock(); _play2dMutex.unlock(); if (_streamingSource) { if (_rotateBufferThread != nullptr) { while (!_isRotateThreadExited) { _sleepCondition.notify_one(); std::this_thread::sleep_for(std::chrono::milliseconds(5)); } if (_rotateBufferThread->joinable()) { _rotateBufferThread->join(); } delete _rotateBufferThread; _rotateBufferThread = nullptr; ALOGVV("rotateBufferThread exited!"); #if CC_TARGET_PLATFORM == CC_PLATFORM_IOS // some specific OpenAL implement defects existed on iOS platform // refer to: https://github.com/cocos2d/cocos2d-x/issues/18597 ALint sourceState; ALint bufferProcessed = 0; alGetSourcei(_alSource, AL_SOURCE_STATE, &sourceState); if (sourceState == AL_PLAYING) { alGetSourcei(_alSource, AL_BUFFERS_PROCESSED, &bufferProcessed); while (bufferProcessed < QUEUEBUFFER_NUM) { std::this_thread::sleep_for(std::chrono::milliseconds(2)); alGetSourcei(_alSource, AL_BUFFERS_PROCESSED, &bufferProcessed); } alSourceUnqueueBuffers(_alSource, QUEUEBUFFER_NUM, _bufferIds); CHECK_AL_ERROR_DEBUG(); } ALOGVV("UnqueueBuffers Before alSourceStop"); #endif } } } while (false); ALOGVV("Before alSourceStop"); alSourceStop(_alSource); CHECK_AL_ERROR_DEBUG(); ALOGVV("Before alSourcei"); alSourcei(_alSource, AL_BUFFER, 0); CHECK_AL_ERROR_DEBUG(); _removeByAudioEngine = true; _ready = false; ALOGVV("AudioPlayer::destroy end, id=%u", _id); } void AudioPlayer::setCache(AudioCache *cache) { _audioCache = cache; } bool AudioPlayer::play2d() { _play2dMutex.lock(); ALOGVV("AudioPlayer::play2d, _alSource: %u", _alSource); /*********************************************************************/ /* Note that it may be in sub thread or in main thread. **/ /*********************************************************************/ bool ret = false; do { if (_audioCache->_state != AudioCache::State::READY) { ALOGE("alBuffer isn't ready for play!"); break; } alSourcei(_alSource, AL_BUFFER, 0); CHECK_AL_ERROR_DEBUG(); alSourcef(_alSource, AL_PITCH, 1.0f); CHECK_AL_ERROR_DEBUG(); alSourcef(_alSource, AL_GAIN, _volume); CHECK_AL_ERROR_DEBUG(); alSourcei(_alSource, AL_LOOPING, AL_FALSE); CHECK_AL_ERROR_DEBUG(); if (_audioCache->_queBufferFrames == 0) { if (_loop) { alSourcei(_alSource, AL_LOOPING, AL_TRUE); CHECK_AL_ERROR_DEBUG(); } } else { if (_currTime > _audioCache->_duration) { _currTime = 0.F; // Target current start time is invalid, reset to 0. } alGenBuffers(QUEUEBUFFER_NUM, _bufferIds); auto alError = alGetError(); if (alError == AL_NO_ERROR) { for (int index = 0; index < QUEUEBUFFER_NUM; ++index) { alBufferData(_bufferIds[index], _audioCache->_format, _audioCache->_queBuffers[index], _audioCache->_queBufferSize[index], _audioCache->_sampleRate); } CHECK_AL_ERROR_DEBUG(); } else { ALOGE("%s:alGenBuffers error code:%x", __PRETTY_FUNCTION__, alError); break; } _streamingSource = true; } { std::unique_lock lk(_sleepMutex); if (_isDestroyed) break; if (_streamingSource) { // To continuously stream audio from a source without interruption, buffer queuing is required. alSourceQueueBuffers(_alSource, QUEUEBUFFER_NUM, _bufferIds); CHECK_AL_ERROR_DEBUG(); _rotateBufferThread = ccnew std::thread(&AudioPlayer::rotateBufferThread, this, _audioCache->_queBufferFrames * QUEUEBUFFER_NUM + 1); } else { alSourcei(_alSource, AL_BUFFER, _audioCache->_alBufferId); CHECK_AL_ERROR_DEBUG(); } alSourcePlay(_alSource); } auto alError = alGetError(); if (alError != AL_NO_ERROR) { ALOGE("%s:alSourcePlay error code:%x", __PRETTY_FUNCTION__, alError); break; } /** Due to the bug of OpenAL, when the second time OpenAL trying to mix audio into bus, the mRampState become kRampingComplete, and for those oalSource whose mRampState == kRampingComplete, nothing happens. * OALSource::Play{ * switch(mState){ * case kTransitionToStop: * case kTransitionToStop: * if(mRampState != kRampingComplete){..} * break; * } * } * So the assert here will trigger this bug as aolSource is reused. * Replace OpenAL with AVAudioEngine on V3.6 mightbe helpful */ // CC_ASSERT_EQ(state, AL_PLAYING); _ready = true; ret = true; } while (false); if (!ret) { _removeByAudioEngine = true; } _play2dMutex.unlock(); return ret; } // rotateBufferThread is used to rotate alBufferData for _alSource when playing big audio file void AudioPlayer::rotateBufferThread(int offsetFrame) { char *tmpBuffer = nullptr; AudioDecoder decoder; long long rotateSleepTime = static_cast(QUEUEBUFFER_TIME_STEP * 1000) / 2; do { BREAK_IF(!decoder.open(_audioCache->_fileFullPath.c_str())); uint32_t framesRead = 0; const uint32_t framesToRead = _audioCache->_queBufferFrames; const uint32_t bufferSize = framesToRead * decoder.getBytesPerFrame(); tmpBuffer = (char *)malloc(bufferSize); memset(tmpBuffer, 0, bufferSize); if (offsetFrame != 0) { decoder.seek(offsetFrame); } ALint sourceState; ALint bufferProcessed = 0; bool needToExitThread = false; while (!_isDestroyed) { alGetSourcei(_alSource, AL_SOURCE_STATE, &sourceState); /* On IOS, audio state will lie, when the system is not fully foreground, * openAl will process the buffer in queue, but our condition cannot make sure that the audio * is playing as it's too short. Interesting IOS system. * Solution is to load buffer even if it's paused, just make sure that there's no bufferProcessed in */ if (sourceState == AL_PLAYING || sourceState == AL_PAUSED) { alGetSourcei(_alSource, AL_BUFFERS_PROCESSED, &bufferProcessed); while (bufferProcessed > 0) { bufferProcessed--; if (_timeDirty) { _timeDirty = false; offsetFrame = _currTime * decoder.getSampleRate(); decoder.seek(offsetFrame); } else { _currTime += QUEUEBUFFER_TIME_STEP; if (_currTime > _audioCache->_duration) { if (_loop) { _currTime = 0.0f; } else { _currTime = _audioCache->_duration; } } } framesRead = decoder.readFixedFrames(framesToRead, tmpBuffer); if (framesRead == 0) { if (_loop) { decoder.seek(0); framesRead = decoder.readFixedFrames(framesToRead, tmpBuffer); } else { needToExitThread = true; break; } } /* While the source is playing, alSourceUnqueueBuffers can be called to remove buffers which have already played. Those buffers can then be filled with new data or discarded. New or refilled buffers can then be attached to the playing source using alSourceQueueBuffers. As long as there is always a new buffer to play in the queue, the source will continue to play. */ ALuint bid; alSourceUnqueueBuffers(_alSource, 1, &bid); alBufferData(bid, _audioCache->_format, tmpBuffer, framesRead * decoder.getBytesPerFrame(), decoder.getSampleRate()); alSourceQueueBuffers(_alSource, 1, &bid); } } std::unique_lock lk(_sleepMutex); if (_isDestroyed || needToExitThread) { break; } if (!_needWakeupRotateThread) { _sleepCondition.wait_for(lk, std::chrono::milliseconds(rotateSleepTime)); } _needWakeupRotateThread = false; } } while (false); ALOGVV("Exit rotate buffer thread ..."); decoder.close(); free(tmpBuffer); _isRotateThreadExited = true; } void AudioPlayer::wakeupRotateThread() { _needWakeupRotateThread = true; _sleepCondition.notify_all(); } bool AudioPlayer::setLoop(bool loop) { if (!_isDestroyed) { _loop = loop; return true; } return false; } bool AudioPlayer::setTime(float time) { if (!_isDestroyed && time >= 0.0f && time < _audioCache->_duration) { _currTime = time; _timeDirty = true; return true; } return false; }