2022-01-03 11:42:07 +08:00
|
|
|
/****************************************************************************
|
|
|
|
Copyright (c) 2016 Chukong Technologies Inc.
|
|
|
|
Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd.
|
2022-04-25 19:15:46 +08:00
|
|
|
Copyright (c) 2020 C4games Ltd
|
2022-01-03 11:42:07 +08:00
|
|
|
Copyright (c) 2021 Bytedance Inc.
|
|
|
|
|
2022-10-01 16:24:52 +08:00
|
|
|
https://axmolengine.github.io/
|
2022-01-03 11:42:07 +08:00
|
|
|
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
|
|
in the Software without restriction, including without limitation the rights
|
|
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
|
|
all copies or substantial portions of the Software.
|
|
|
|
|
|
|
|
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 "AudioDecoderMp3"
|
|
|
|
#include "audio/AudioDecoderMp3.h"
|
|
|
|
#include "audio/AudioMacros.h"
|
2023-06-11 13:08:08 +08:00
|
|
|
#include "platform/FileUtils.h"
|
2022-01-03 11:42:07 +08:00
|
|
|
|
2023-06-11 13:08:08 +08:00
|
|
|
#include "base/Console.h"
|
2022-01-03 11:42:07 +08:00
|
|
|
|
2022-07-16 10:43:05 +08:00
|
|
|
#if !AX_USE_MPG123
|
2022-01-03 11:42:07 +08:00
|
|
|
# define MINIMP3_IMPLEMENTATION
|
|
|
|
# include "minimp3/minimp3_ex.h"
|
|
|
|
#else
|
|
|
|
// because the cocos2d-x engine has ssize_t typedef, so disable mpg123 ssize_t
|
|
|
|
# define MPG123_DEF_SSIZE_T
|
|
|
|
# include "mpg123.h"
|
|
|
|
#endif
|
|
|
|
|
2022-07-16 10:43:05 +08:00
|
|
|
#if AX_TARGET_PLATFORM == AX_PLATFORM_ANDROID
|
2022-01-03 11:42:07 +08:00
|
|
|
# include <unistd.h>
|
|
|
|
# include <errno.h>
|
|
|
|
#endif
|
|
|
|
|
2022-07-16 10:43:05 +08:00
|
|
|
#if !AX_USE_MPG123
|
2022-01-03 11:42:07 +08:00
|
|
|
struct mp3dec_impl
|
|
|
|
{
|
|
|
|
mp3dec_ex_t _dec;
|
|
|
|
mp3dec_io_t _decIO;
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
2022-07-11 17:50:21 +08:00
|
|
|
NS_AX_BEGIN
|
|
|
|
|
2022-07-16 10:43:05 +08:00
|
|
|
#if !AX_USE_MPG123
|
2022-01-03 11:42:07 +08:00
|
|
|
static size_t minimp3_read_r(void* buf, size_t size, void* user_data)
|
|
|
|
{
|
|
|
|
return ((FileStream*)user_data)->read(buf, size);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int minimp3_seek_r(uint64_t position, void* user_data)
|
|
|
|
{
|
|
|
|
return ((FileStream*)user_data)->seek(position, SEEK_SET) < 0 ? -1 : 0;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
static bool __mp3Inited = false;
|
|
|
|
|
|
|
|
static ssize_t mpg123_read_r(void* handle, void* buffer, size_t count)
|
|
|
|
{
|
|
|
|
return ((FileStream*)handle)->read(buffer, count);
|
|
|
|
}
|
|
|
|
|
|
|
|
static off_t mpg123_lseek_r(void* handle, off_t offset, int whence)
|
|
|
|
{
|
|
|
|
return ((FileStream*)handle)->seek(offset, whence);
|
|
|
|
}
|
|
|
|
|
|
|
|
void mpg123_close_r(void* handle)
|
|
|
|
{
|
|
|
|
((FileStream*)handle)->close();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
bool AudioDecoderMp3::lazyInit()
|
|
|
|
{
|
|
|
|
bool ret = true;
|
2022-07-16 10:43:05 +08:00
|
|
|
#if AX_USE_MPG123
|
2022-01-03 11:42:07 +08:00
|
|
|
if (!__mp3Inited)
|
|
|
|
{
|
|
|
|
int error = mpg123_init();
|
|
|
|
if (error == MPG123_OK)
|
|
|
|
{
|
|
|
|
__mp3Inited = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ALOGE("Basic setup goes wrong: %s", mpg123_plain_strerror(error));
|
|
|
|
ret = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AudioDecoderMp3::destroy()
|
|
|
|
{
|
2022-07-16 10:43:05 +08:00
|
|
|
#if AX_USE_MPG123
|
2022-01-03 11:42:07 +08:00
|
|
|
if (__mp3Inited)
|
|
|
|
{
|
|
|
|
mpg123_exit();
|
|
|
|
__mp3Inited = false;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
AudioDecoderMp3::AudioDecoderMp3() : _handle(nullptr)
|
|
|
|
{
|
|
|
|
lazyInit();
|
|
|
|
}
|
|
|
|
|
|
|
|
AudioDecoderMp3::~AudioDecoderMp3()
|
|
|
|
{
|
|
|
|
close();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AudioDecoderMp3::open(std::string_view fullPath)
|
|
|
|
{
|
2022-07-16 10:43:05 +08:00
|
|
|
#if !AX_USE_MPG123
|
2022-01-03 11:42:07 +08:00
|
|
|
do
|
|
|
|
{
|
|
|
|
_fileStream = FileUtils::getInstance()->openFileStream(fullPath, FileStream::Mode::READ);
|
|
|
|
if (!_fileStream)
|
|
|
|
{
|
|
|
|
ALOGE("Trouble with minimp3(1): %s\n", strerror(errno));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto handle = new mp3dec_impl();
|
|
|
|
|
|
|
|
handle->_decIO.read = minimp3_read_r;
|
|
|
|
handle->_decIO.seek = minimp3_seek_r;
|
|
|
|
handle->_decIO.read_data = handle->_decIO.seek_data = _fileStream.get();
|
|
|
|
|
|
|
|
if (mp3dec_ex_open_cb(&handle->_dec, &handle->_decIO, MP3D_SEEK_TO_SAMPLE) != 0)
|
|
|
|
{
|
|
|
|
delete handle;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto& info = handle->_dec.info;
|
|
|
|
_channelCount = info.channels; // the _channelCount is samplesPerFrame
|
|
|
|
_sampleRate = info.hz;
|
|
|
|
# ifndef MINIMP3_FLOAT_OUTPUT
|
|
|
|
_sourceFormat = AUDIO_SOURCE_FORMAT::PCM_16;
|
|
|
|
_bytesPerBlock = sizeof(uint16_t) * _channelCount;
|
|
|
|
# else
|
|
|
|
_sourceFormat = AUDIO_SOURCE_FORMAT::PCM_FLT32;
|
|
|
|
_bytesPerBlock = sizeof(float) * _channelCount;
|
|
|
|
# endif
|
|
|
|
// samples
|
|
|
|
_totalFrames = handle->_dec.samples / _channelCount;
|
|
|
|
|
|
|
|
// store the handle
|
|
|
|
_handle = handle;
|
|
|
|
|
|
|
|
_isOpened = true;
|
|
|
|
return true;
|
|
|
|
} while (false);
|
|
|
|
|
|
|
|
return false;
|
|
|
|
#else
|
|
|
|
int32_t rate = 0;
|
|
|
|
int error = MPG123_OK;
|
|
|
|
int mp3Encoding = 0;
|
|
|
|
int channel = 0;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
_handle = mpg123_new(nullptr, &error);
|
|
|
|
if (nullptr == _handle)
|
|
|
|
{
|
|
|
|
ALOGE("Basic setup goes wrong: %s", mpg123_plain_strerror(error));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!_fileStream.open(fullPath))
|
|
|
|
{
|
|
|
|
ALOGE("Trouble with mpg123(1): %s\n", strerror(errno));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
mpg123_replace_reader_handle(_handle, mpg123_read_r, mpg123_lseek_r, mpg123_close_r);
|
|
|
|
|
|
|
|
if (mpg123_open_handle(_handle, _fileStream) != MPG123_OK ||
|
|
|
|
mpg123_getformat(_handle, &rate, &channel, &mp3Encoding) != MPG123_OK)
|
|
|
|
{
|
|
|
|
ALOGE("Trouble with mpg123(2): %s\n", mpg123_strerror(_handle));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
_channelCount = channel;
|
|
|
|
_sampleRate = rate;
|
|
|
|
|
|
|
|
if (mp3Encoding == MPG123_ENC_SIGNED_16)
|
|
|
|
{
|
|
|
|
_bytesPerBlock = 2 * _channelCount;
|
|
|
|
_sourceFormat = AUDIO_SOURCE_FORMAT::PCM_16;
|
|
|
|
}
|
|
|
|
else if (mp3Encoding == MPG123_ENC_FLOAT_32)
|
|
|
|
{
|
|
|
|
_bytesPerBlock = 4 * _channelCount;
|
|
|
|
_sourceFormat = AUDIO_SOURCE_FORMAT::PCM_FLT32;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ALOGE("Bad encoding: 0x%x!\n", mp3Encoding);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Ensure that this output format will not change (it could, when we allow it). */
|
|
|
|
mpg123_format_none(_handle);
|
|
|
|
mpg123_format(_handle, rate, channel, mp3Encoding);
|
|
|
|
/* Ensure that we can get accurate length by call mpg123_length */
|
|
|
|
mpg123_scan(_handle);
|
|
|
|
|
|
|
|
_totalFrames = mpg123_length(_handle);
|
|
|
|
|
|
|
|
_isOpened = true;
|
|
|
|
return true;
|
|
|
|
} while (false);
|
|
|
|
|
|
|
|
if (_handle != nullptr)
|
|
|
|
{
|
|
|
|
mpg123_close(_handle);
|
|
|
|
mpg123_delete(_handle);
|
|
|
|
_handle = nullptr;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void AudioDecoderMp3::close()
|
|
|
|
{
|
|
|
|
if (isOpened())
|
|
|
|
{
|
2022-07-16 10:43:05 +08:00
|
|
|
#if !AX_USE_MPG123
|
2022-01-03 11:42:07 +08:00
|
|
|
if (_handle)
|
|
|
|
{
|
|
|
|
mp3dec_ex_close(&_handle->_dec);
|
|
|
|
|
|
|
|
delete _handle;
|
|
|
|
_handle = nullptr;
|
|
|
|
_fileStream.reset();
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
if (_handle != nullptr)
|
|
|
|
{
|
|
|
|
mpg123_close(_handle);
|
|
|
|
mpg123_delete(_handle);
|
|
|
|
_handle = nullptr;
|
|
|
|
|
|
|
|
_fileStream.reset();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
_isOpened = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t AudioDecoderMp3::read(uint32_t framesToRead, char* pcmBuf)
|
|
|
|
{
|
2022-07-16 10:43:05 +08:00
|
|
|
#if !AX_USE_MPG123
|
2022-01-03 11:42:07 +08:00
|
|
|
auto sampelsToRead = framesToRead * _channelCount;
|
|
|
|
auto samplesRead = mp3dec_ex_read(&_handle->_dec, (mp3d_sample_t*)pcmBuf, sampelsToRead);
|
|
|
|
return static_cast<uint32_t>(samplesRead / _channelCount);
|
|
|
|
#else
|
|
|
|
int bytesToRead = framesToBytes(framesToRead);
|
|
|
|
size_t bytesRead = 0;
|
|
|
|
int err = mpg123_read(_handle, (unsigned char*)pcmBuf, bytesToRead, &bytesRead);
|
|
|
|
if (err == MPG123_ERR)
|
|
|
|
{
|
|
|
|
ALOGE("Trouble with mpg123: %s\n", mpg123_strerror(_handle));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return bytesToFrames(bytesRead);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AudioDecoderMp3::seek(uint32_t frameOffset)
|
|
|
|
{
|
2022-07-16 10:43:05 +08:00
|
|
|
#if !AX_USE_MPG123
|
2022-01-03 11:42:07 +08:00
|
|
|
return 0 == mp3dec_ex_seek(&_handle->_dec, frameOffset);
|
|
|
|
#else
|
|
|
|
off_t offset = mpg123_seek(_handle, frameOffset, SEEK_SET);
|
|
|
|
// ALOGD("mpg123_seek return: %d", (int)offset);
|
|
|
|
return (offset >= 0 && offset == frameOffset);
|
|
|
|
#endif
|
|
|
|
}
|
2022-08-29 20:51:22 +08:00
|
|
|
NS_AX_END // namespace ax
|