openspades/Sources/Core/WavAudioStream.cpp

124 lines
3.3 KiB
C++
Raw Normal View History

2013-08-29 11:45:22 +09:00
/*
Copyright (c) 2013 yvt
2013-08-29 11:45:22 +09:00
This file is part of OpenSpades.
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.
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.
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/>.
2013-08-29 11:45:22 +09:00
*/
2013-08-18 16:18:06 +09:00
#include "WavAudioStream.h"
#include "Debug.h"
#include "Exception.h"
2013-08-18 16:18:06 +09:00
namespace spades {
WavAudioStream::WavAudioStream(IStream *s, bool ac) {
SPADES_MARK_FUNCTION();
2013-08-18 16:18:06 +09:00
stream = s;
autoClose = ac;
2013-08-18 16:18:06 +09:00
// skip header
s->SetPosition(12 + s->GetPosition());
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);
}
const RiffChunkInfo &fmt = GetChunk("fmt ");
2013-08-18 16:18:06 +09:00
stream->SetPosition(fmt.dataPosition);
stream->ReadLittleShort(); // ??
2013-08-18 16:18:06 +09:00
channels = stream->ReadLittleShort();
rate = stream->ReadLittleInt();
stream->ReadLittleInt();
stream->ReadLittleShort();
int bits = stream->ReadLittleShort();
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
}
2013-08-18 16:18:06 +09:00
dataChunk = &GetChunk("data");
stream->SetPosition(dataChunk->dataPosition);
2013-08-18 16:18:06 +09:00
startPos = dataChunk->dataPosition;
endPos = dataChunk->dataPosition + dataChunk->length;
}
const WavAudioStream::RiffChunkInfo &WavAudioStream::GetChunk(const std::string &name) {
2013-08-18 16:18:06 +09:00
SPADES_MARK_FUNCTION();
2013-08-18 16:18:06 +09:00
std::map<std::string, RiffChunkInfo>::iterator it;
it = chunks.find(name);
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;
}
2013-08-18 16:18:06 +09:00
WavAudioStream::RiffChunkInfo WavAudioStream::ReadChunkInfo() {
SPADES_MARK_FUNCTION();
2013-08-18 16:18:06 +09:00
RiffChunkInfo info;
info.name = stream->Read(4);
if (info.name.size() < 4)
2013-08-18 16:18:06 +09:00
SPRaise("Failed to read RIFF header name");
2013-08-18 16:18:06 +09:00
info.length = stream->ReadLittleInt();
info.dataPosition = stream->GetPosition();
return info;
}
WavAudioStream::~WavAudioStream() {
2013-08-18 16:18:06 +09:00
SPADES_MARK_FUNCTION();
if (autoClose)
2013-08-18 16:18:06 +09:00
delete stream;
}
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();
if (stream->GetPosition() >= endPos)
2013-08-18 16:18:06 +09:00
return -1;
else
return stream->ReadByte();
}
2013-08-18 16:18:06 +09:00
size_t WavAudioStream::Read(void *data, size_t bytes) {
SPADES_MARK_FUNCTION();
2013-08-18 16:18:06 +09:00
uint64_t maxLen = endPos - stream->GetPosition();
if ((uint64_t)bytes > maxLen)
2013-08-18 16:18:06 +09:00
bytes = (size_t)maxLen;
return stream->Read(data, bytes);
}
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
}