/*************************************************************************** copyright : (C) 2006 by Lukáš Lalinský email : lalinsky@gmail.com copyright : (C) 2004 by Allan Sandfeld Jensen email : kde@carewolf.org (original MPC implementation) ***************************************************************************/ /*************************************************************************** * This library is free software; you can redistribute it and/or modify * * it under the terms of the GNU Lesser General Public License version * * 2.1 as published by the Free Software Foundation. * * * * This library 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 * * Lesser General Public License for more details. * * * * You should have received a copy of the GNU Lesser General Public * * License along with this library; if not, write to the Free Software * * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA * * 02110-1301 USA * * * * Alternatively, this file is available under the Mozilla Public * * License Version 1.1. You may obtain a copy of the License at * * http://www.mozilla.org/MPL/ * ***************************************************************************/ #include #include #include "wavpackproperties.h" #include "wavpackfile.h" // Implementation of this class is based on the information at: // http://www.wavpack.com/file_format.txt using namespace TagLib; class WavPack::Properties::PropertiesPrivate { public: PropertiesPrivate() : length(0), bitrate(0), sampleRate(0), channels(0), version(0), bitsPerSample(0), lossless(false), sampleFrames(0) {} int length; int bitrate; int sampleRate; int channels; int version; int bitsPerSample; bool lossless; unsigned int sampleFrames; }; //////////////////////////////////////////////////////////////////////////////// // public members //////////////////////////////////////////////////////////////////////////////// WavPack::Properties::Properties(const ByteVector &, long, ReadStyle style) : AudioProperties(style), d(new PropertiesPrivate()) { debug("WavPack::Properties::Properties() -- This constructor is no longer used."); } WavPack::Properties::Properties(File *file, long streamLength, ReadStyle style) : AudioProperties(style), d(new PropertiesPrivate()) { read(file, streamLength); } WavPack::Properties::~Properties() { delete d; } int WavPack::Properties::length() const { return lengthInSeconds(); } int WavPack::Properties::lengthInSeconds() const { return d->length / 1000; } int WavPack::Properties::lengthInMilliseconds() const { return d->length; } int WavPack::Properties::bitrate() const { return d->bitrate; } int WavPack::Properties::sampleRate() const { return d->sampleRate; } int WavPack::Properties::channels() const { return d->channels; } int WavPack::Properties::version() const { return d->version; } int WavPack::Properties::bitsPerSample() const { return d->bitsPerSample; } bool WavPack::Properties::isLossless() const { return d->lossless; } unsigned int WavPack::Properties::sampleFrames() const { return d->sampleFrames; } //////////////////////////////////////////////////////////////////////////////// // private members //////////////////////////////////////////////////////////////////////////////// namespace { const unsigned int sample_rates[] = { 6000, 8000, 9600, 11025, 12000, 16000, 22050, 24000, 32000, 44100, 48000, 64000, 88200, 96000, 192000, 0 }; } #define BYTES_STORED 3 #define MONO_FLAG 4 #define LOSSLESS_FLAG 8 #define SHIFT_LSB 13 #define SHIFT_MASK (0x1fL << SHIFT_LSB) #define SRATE_LSB 23 #define SRATE_MASK (0xfL << SRATE_LSB) #define MIN_STREAM_VERS 0x402 #define MAX_STREAM_VERS 0x410 #define FINAL_BLOCK 0x1000 void WavPack::Properties::read(File *file, long streamLength) { long offset = 0; while(true) { file->seek(offset); const ByteVector data = file->readBlock(32); if(data.size() < 32) { debug("WavPack::Properties::read() -- data is too short."); break; } if(!data.startsWith("wvpk")) { debug("WavPack::Properties::read() -- Block header not found."); break; } const unsigned int flags = data.toUInt(24, false); if(offset == 0) { d->version = data.toShort(8, false); if(d->version < MIN_STREAM_VERS || d->version > MAX_STREAM_VERS) break; d->bitsPerSample = ((flags & BYTES_STORED) + 1) * 8 - ((flags & SHIFT_MASK) >> SHIFT_LSB); d->sampleRate = sample_rates[(flags & SRATE_MASK) >> SRATE_LSB]; d->lossless = !(flags & LOSSLESS_FLAG); d->sampleFrames = data.toUInt(12, false); } d->channels += (flags & MONO_FLAG) ? 1 : 2; if(flags & FINAL_BLOCK) break; const unsigned int blockSize = data.toUInt(4, false); offset += blockSize + 8; } if(d->sampleFrames == ~0u) d->sampleFrames = seekFinalIndex(file, streamLength); if(d->sampleFrames > 0 && d->sampleRate > 0) { const double length = d->sampleFrames * 1000.0 / d->sampleRate; d->length = static_cast(length + 0.5); d->bitrate = static_cast(streamLength * 8.0 / length + 0.5); } } unsigned int WavPack::Properties::seekFinalIndex(File *file, long streamLength) { const long offset = file->rfind("wvpk", streamLength); if(offset == -1) return 0; file->seek(offset); const ByteVector data = file->readBlock(32); if(data.size() < 32) return 0; const int version = data.toShort(8, false); if(version < MIN_STREAM_VERS || version > MAX_STREAM_VERS) return 0; const unsigned int flags = data.toUInt(24, false); if(!(flags & FINAL_BLOCK)) return 0; const unsigned int blockIndex = data.toUInt(16, false); const unsigned int blockSamples = data.toUInt(20, false); return blockIndex + blockSamples; }