/***************************************************************************
* Copyright (C) 2008-2019 by Ilya Kotov *
* forkotov02@ya.ru *
* *
* This program is free software; you can redistribute it and/or modify *
* it under the terms of the GNU General Public License as published by *
* the Free Software Foundation; either version 2 of the License, or *
* (at your option) any later version. *
* *
* This program is distributed in the hope that it will be useful, *
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
* GNU General Public License for more details. *
* *
* You should have received a copy of the GNU General Public License *
* along with this program; if not, write to the *
* Free Software Foundation, Inc., *
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *
***************************************************************************/
#include <QObject>
#include <QIODevice>
#include <QFile>
#include <QRegExp>
#include <math.h>
#include <stdint.h>
#include <qmmp/cueparser.h>
#include <qmmp/buffer.h>
#include <stdlib.h>
#include "decoder_wavpack.h"
#include "cueparser.h"
// Decoder class
DecoderWavPack::DecoderWavPack(const QString &path)
: Decoder()
{
m_path = path;
m_totalTime = 0.0;
m_chan = 0;
m_context = nullptr;
m_parser = nullptr;
m_output_buf = nullptr;
m_length_in_bytes = 0;
m_totalBytes = 0;
m_frame_size = 0;
m_offset = 0;
m_bps = 0;
m_track = 0;
}
DecoderWavPack::~DecoderWavPack()
{
deinit();
if (m_output_buf)
delete [] m_output_buf;
m_output_buf = nullptr;
}
bool DecoderWavPack::initialize()
{
m_chan = 0;
m_totalTime = 0;
char err[80] = { 0 };
if (m_path.startsWith("wvpack://")) //embeded cue track
{
QString p = m_path;
p.remove("wvpack://");
p.remove(QRegExp("#\\d+$"));
#if defined(Q_OS_WIN) && defined(OPEN_FILE_UTF8)
m_context = WavpackOpenFileInput (p.toUtf8().constData(),
err, OPEN_WVC | OPEN_TAGS | OPEN_FILE_UTF8, 0);
#else
m_context = WavpackOpenFileInput (p.toLocal8Bit().constData(), err, OPEN_WVC | OPEN_TAGS, 0);
#endif
if (!m_context)
{
qWarning("DecoderWavPack: error: %s", err);
return false;
}
int cue_len = WavpackGetTagItem (m_context, "cuesheet", nullptr, 0);
if (cue_len > 0)
{
char *value = (char*)malloc (cue_len * 2 + 1);
WavpackGetTagItem(m_context, "cuesheet", value, cue_len + 1);
m_parser = new CueParser(value);
m_parser->setDuration((qint64)WavpackGetNumSamples(m_context) * 1000 / WavpackGetSampleRate(m_context));
m_parser->setUrl("wvpack", p);
m_track = m_path.section("#", -1).toInt();
if(m_track < 1 || m_track > m_parser->count())
{
qWarning("DecoderWavPack: invalid cuesheet comment");
return false;
}
m_path = p;
//send metadata
QMap<Qmmp::MetaData, QString> metaData = m_parser->info(m_track)->metaData();
addMetaData(metaData);
}
}
else
#if defined(Q_OS_WIN) && defined(OPEN_FILE_UTF8)
m_context = WavpackOpenFileInput (m_path.toUtf8().constData(),
err, OPEN_WVC | OPEN_TAGS | OPEN_FILE_UTF8, 0);
#else
m_context = WavpackOpenFileInput (m_path.toLocal8Bit().constData(), err, OPEN_WVC | OPEN_TAGS, 0);
#endif
if (!m_context)
{
qWarning("DecoderWavPack: error: %s", err);
return false;
}
m_chan = WavpackGetNumChannels(m_context);
uint32_t freq = WavpackGetSampleRate (m_context);
m_bps = WavpackGetBitsPerSample (m_context);
ChannelMap chmap = findChannelMap(m_chan);
if(chmap.isEmpty())
{
qWarning("DecoderWavPack: unsupported number of channels: %d", m_chan);
return false;
}
if (!m_output_buf)
m_output_buf = new int32_t[QMMP_BLOCK_FRAMES * m_chan];
switch(m_bps)
{
case 8:
configure(freq, chmap, Qmmp::PCM_S8);
break;
case 12:
case 16:
configure(freq, chmap, Qmmp::PCM_S16LE);
break;
case 20:
case 24:
case 32:
#ifdef MODE_FLOAT
configure(freq, chmap, (WavpackGetMode(m_context) & MODE_FLOAT) ? Qmmp::PCM_FLOAT : Qmmp::PCM_S32LE);
#else
configure(freq, chmap, Qmmp::PCM_S32LE);
#endif
break;
default:
qWarning("DecoderWavPack: unsupported bit depth");
return false;
}
if(!m_parser)
m_totalTime = (qint64) WavpackGetNumSamples(m_context) * 1000 / freq;
else
{
m_length = m_parser->duration(m_track);
m_offset = m_parser->offset(m_track);
m_length_in_bytes = audioParameters().sampleRate() *
audioParameters().frameSize() * m_length/1000;
setReplayGainInfo(m_parser->info(m_track)->replayGainInfo());
seek(0);
}
m_totalBytes = 0;
m_frame_size = audioParameters().frameSize();
qDebug("DecoderWavPack: initialize succes");
return true;
}
int DecoderWavPack::bitrate() const
{
if(m_context)
return int(WavpackGetInstantBitrate(m_context)/1000);
return 0;
}
qint64 DecoderWavPack::totalTime() const
{
if(m_parser)
return m_length;
return m_totalTime;
}
void DecoderWavPack::deinit()
{
m_chan = 0;
if (m_context)
WavpackCloseFile (m_context);
m_context = nullptr;
if(m_parser)
delete m_parser;
m_parser = nullptr;
}
void DecoderWavPack::seek(qint64 time)
{
m_totalBytes = audioParameters().sampleRate() *
audioParameters().channels() *
audioParameters().sampleSize() * time/1000;
if(m_parser)
time += m_offset;
WavpackSeekSample (m_context, time * audioParameters().sampleRate() / 1000);
}
qint64 DecoderWavPack::read(unsigned char *data, qint64 size)
{
if(m_parser)
{
if(m_length_in_bytes - m_totalBytes < m_frame_size) //end of cue track
return 0;
//returned size must contain integer number of frames
size = qMin(size, (m_length_in_bytes - m_totalBytes) / m_frame_size * m_frame_size);
size = wavpack_decode(data, size);
m_totalBytes += size;
return size;
}
return wavpack_decode(data, size);
}
const QString DecoderWavPack::nextURL() const
{
if(m_parser && m_track +1 <= m_parser->count())
return m_parser->url(m_track + 1);
else
return QString();
}
void DecoderWavPack::next()
{
if(m_parser && m_track +1 <= m_parser->count())
{
m_track++;
m_offset = m_parser->duration(m_track);
m_length = m_parser->duration(m_track);
m_length_in_bytes = audioParameters().sampleRate() *
audioParameters().channels() *
audioParameters().sampleSize() * m_length/1000;
addMetaData(m_parser->info(m_track)->metaData());
setReplayGainInfo(m_parser->info(m_track)->replayGainInfo());
m_totalBytes = 0;
}
}
qint64 DecoderWavPack::wavpack_decode(unsigned char *data, qint64 size)
{
ulong len = qMin(QMMP_BLOCK_FRAMES, (int)size / m_chan / 4);
len = WavpackUnpackSamples (m_context, m_output_buf, len);
//convert 32 to 16
qint8 *data8 = (qint8 *)data;
qint16 *data16 = (qint16 *)data;
qint32 *data32 = (qint32 *)data;
uint i = 0;
switch (m_bps)
{
case 8:
for (i = 0; i < len * m_chan; ++i)
data8[i] = m_output_buf[i];
return len * m_chan;
case 12:
case 16:
for (i = 0; i < len * m_chan; ++i)
data16[i] = m_output_buf[i];
return len * m_chan * 2;
case 20:
case 24:
for (i = 0; i < len * m_chan; ++i)
data32[i] = m_output_buf[i] << 8;
return len * m_chan * 4;
case 32:
for (i = 0; i < len * m_chan; ++i)
data32[i] = m_output_buf[i];
return len * m_chan * 4;
}
return 0;
}
//http://www.wavpack.com/file_format.txt
ChannelMap DecoderWavPack::findChannelMap(int channels)
{
ChannelMap map;
switch (channels)
{
case 1:
map << Qmmp::CHAN_FRONT_LEFT;
break;
case 2:
map << Qmmp::CHAN_FRONT_LEFT
<< Qmmp::CHAN_FRONT_RIGHT;
break;
case 3:
map << Qmmp::CHAN_FRONT_LEFT
<< Qmmp::CHAN_FRONT_RIGHT
<< Qmmp::CHAN_FRONT_CENTER;
break;
case 4:
map << Qmmp::CHAN_FRONT_LEFT
<< Qmmp::CHAN_FRONT_RIGHT
<< Qmmp::CHAN_REAR_LEFT
<< Qmmp::CHAN_REAR_RIGHT;
break;
case 5:
map << Qmmp::CHAN_FRONT_LEFT
<< Qmmp::CHAN_FRONT_RIGHT
<< Qmmp::CHAN_FRONT_CENTER
<< Qmmp::CHAN_REAR_LEFT
<< Qmmp::CHAN_REAR_RIGHT;
break;
case 6:
map << Qmmp::CHAN_FRONT_LEFT
<< Qmmp::CHAN_FRONT_RIGHT
<< Qmmp::CHAN_FRONT_CENTER
<< Qmmp::CHAN_LFE
<< Qmmp::CHAN_REAR_LEFT
<< Qmmp::CHAN_REAR_RIGHT;
break;
case 7:
map << Qmmp::CHAN_FRONT_LEFT
<< Qmmp::CHAN_FRONT_RIGHT
<< Qmmp::CHAN_FRONT_CENTER
<< Qmmp::CHAN_LFE
<< Qmmp::CHAN_REAR_CENTER
<< Qmmp::CHAN_SIDE_LEFT
<< Qmmp::CHAN_SIDE_RIGHT;
break;
case 8:
map << Qmmp::CHAN_FRONT_LEFT
<< Qmmp::CHAN_FRONT_RIGHT
<< Qmmp::CHAN_FRONT_CENTER
<< Qmmp::CHAN_LFE
<< Qmmp::CHAN_REAR_LEFT
<< Qmmp::CHAN_REAR_RIGHT
<< Qmmp::CHAN_SIDE_LEFT
<< Qmmp::CHAN_SIDE_RIGHT;
break;
default:
;
}
return map;
}