2013-08-29 11:45:22 +09:00
|
|
|
/*
|
|
|
|
Copyright (c) 2013 yvt
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-29 11:45:22 +09:00
|
|
|
This file is part of OpenSpades.
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-29 11:45:22 +09:00
|
|
|
OpenSpades 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 3 of the License, or
|
|
|
|
(at your option) any later version.
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-29 11:45:22 +09:00
|
|
|
OpenSpades 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.
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-29 11:45:22 +09:00
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with OpenSpades. If not, see <http://www.gnu.org/licenses/>.
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-29 11:45:22 +09:00
|
|
|
*/
|
2013-08-18 16:18:06 +09:00
|
|
|
|
|
|
|
#include "WavAudioStream.h"
|
|
|
|
#include "Debug.h"
|
2016-12-03 18:23:47 +09:00
|
|
|
#include "Exception.h"
|
2013-08-18 16:18:06 +09:00
|
|
|
|
|
|
|
namespace spades {
|
|
|
|
WavAudioStream::WavAudioStream(IStream *s, bool ac) {
|
|
|
|
SPADES_MARK_FUNCTION();
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
stream = s;
|
|
|
|
autoClose = ac;
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
// skip header
|
|
|
|
s->SetPosition(12 + s->GetPosition());
|
2016-12-03 18:23:47 +09:00
|
|
|
while (s->GetPosition() < s->GetLength()) {
|
2013-08-18 16:18:06 +09:00
|
|
|
RiffChunkInfo info = ReadChunkInfo();
|
|
|
|
chunks[info.name] = info;
|
|
|
|
s->SetPosition(info.dataPosition + info.length);
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
const RiffChunkInfo &fmt = GetChunk("fmt ");
|
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
stream->SetPosition(fmt.dataPosition);
|
2016-12-03 18:23:47 +09:00
|
|
|
stream->ReadLittleShort(); // ??
|
2013-08-18 16:18:06 +09:00
|
|
|
channels = stream->ReadLittleShort();
|
|
|
|
rate = stream->ReadLittleInt();
|
|
|
|
stream->ReadLittleInt();
|
|
|
|
stream->ReadLittleShort();
|
|
|
|
int bits = stream->ReadLittleShort();
|
2016-12-03 18:23:47 +09:00
|
|
|
switch (bits) {
|
|
|
|
case 8: sampleFormat = UnsignedByte; break;
|
|
|
|
case 16: sampleFormat = SignedShort; break;
|
|
|
|
case 32: sampleFormat = SingleFloat; break;
|
|
|
|
default: SPRaise("Unsupported bit count: %d", bits);
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
dataChunk = &GetChunk("data");
|
|
|
|
stream->SetPosition(dataChunk->dataPosition);
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
startPos = dataChunk->dataPosition;
|
|
|
|
endPos = dataChunk->dataPosition + dataChunk->length;
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
const WavAudioStream::RiffChunkInfo &WavAudioStream::GetChunk(const std::string &name) {
|
2013-08-18 16:18:06 +09:00
|
|
|
SPADES_MARK_FUNCTION();
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
std::map<std::string, RiffChunkInfo>::iterator it;
|
|
|
|
it = chunks.find(name);
|
2016-12-03 18:23:47 +09:00
|
|
|
if (it == chunks.end()) {
|
|
|
|
SPRaise("Failed to find RIFF chunk: '%s'", name.c_str());
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
return it->second;
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
WavAudioStream::RiffChunkInfo WavAudioStream::ReadChunkInfo() {
|
|
|
|
SPADES_MARK_FUNCTION();
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
RiffChunkInfo info;
|
|
|
|
info.name = stream->Read(4);
|
2016-12-03 18:23:47 +09:00
|
|
|
if (info.name.size() < 4)
|
2013-08-18 16:18:06 +09:00
|
|
|
SPRaise("Failed to read RIFF header name");
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
info.length = stream->ReadLittleInt();
|
|
|
|
info.dataPosition = stream->GetPosition();
|
|
|
|
return info;
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
WavAudioStream::~WavAudioStream() {
|
2013-08-18 16:18:06 +09:00
|
|
|
SPADES_MARK_FUNCTION();
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
if (autoClose)
|
2013-08-18 16:18:06 +09:00
|
|
|
delete stream;
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
uint64_t WavAudioStream::GetLength() { return dataChunk->length; }
|
|
|
|
|
|
|
|
int WavAudioStream::GetSamplingFrequency() { return rate; }
|
|
|
|
|
|
|
|
int WavAudioStream::GetNumChannels() { return channels; }
|
|
|
|
|
|
|
|
WavAudioStream::SampleFormat WavAudioStream::GetSampleFormat() { return sampleFormat; }
|
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
int WavAudioStream::ReadByte() {
|
|
|
|
SPADES_MARK_FUNCTION();
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
if (stream->GetPosition() >= endPos)
|
2013-08-18 16:18:06 +09:00
|
|
|
return -1;
|
|
|
|
else
|
|
|
|
return stream->ReadByte();
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
size_t WavAudioStream::Read(void *data, size_t bytes) {
|
|
|
|
SPADES_MARK_FUNCTION();
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
uint64_t maxLen = endPos - stream->GetPosition();
|
2016-12-03 18:23:47 +09:00
|
|
|
if ((uint64_t)bytes > maxLen)
|
2013-08-18 16:18:06 +09:00
|
|
|
bytes = (size_t)maxLen;
|
|
|
|
return stream->Read(data, bytes);
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
uint64_t WavAudioStream::GetPosition() { return stream->GetPosition() - startPos; }
|
|
|
|
|
|
|
|
void WavAudioStream::SetPosition(uint64_t pos) { stream->SetPosition(pos + startPos); }
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|