scummvm/audio/decoders/apc.cpp
2022-09-18 12:00:02 +02:00

186 lines
5.3 KiB
C++

/* ScummVM - Graphic Adventure Engine
*
* ScummVM is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
*
* 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 3 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, see <http://www.gnu.org/licenses/>.
*
*/
#include "common/inttypes.h"
#include "common/ptr.h"
#include "common/stream.h"
#include "common/textconsole.h"
#include "common/util.h"
#include "audio/decoders/adpcm_intern.h"
#include "audio/decoders/apc.h"
#include "audio/decoders/raw.h"
namespace Audio {
class APCStreamImpl : public APCStream {
public:
// These parameters are completely optional and only used in HNM videos to make sure data is consistent
APCStreamImpl(uint32 sampleRate = uint(-1), byte stereo = byte(-1));
bool init(Common::SeekableReadStream &header) override;
// AudioStream API
int readBuffer(int16 *buffer, const int numSamples) override { return _audStream->readBuffer(buffer, numSamples); }
bool isStereo() const override { return _audStream->isStereo(); }
int getRate() const override { return _audStream->getRate(); }
bool endOfData() const override { return _audStream->endOfData(); }
bool endOfStream() const override { return _audStream->endOfStream(); }
// PacketizedAudioStream API
void queuePacket(Common::SeekableReadStream *data) override;
void finish() override { _audStream->finish(); }
private:
struct Status {
int32 last;
uint32 stepIndex;
int16 step;
};
FORCEINLINE static int16 decodeIMA(byte code, Status *status);
Common::ScopedPtr<QueuingAudioStream> _audStream;
byte _stereo;
uint32 _sampleRate;
Status _status[2];
};
APCStreamImpl::APCStreamImpl(uint32 sampleRate, byte stereo) :
_sampleRate(sampleRate), _stereo(stereo) {
memset(_status, 0, sizeof(_status));
if (_sampleRate != uint32(-1) && _stereo != byte(-1)) {
_audStream.reset(makeQueuingAudioStream(_sampleRate, _stereo));
}
}
bool APCStreamImpl::init(Common::SeekableReadStream &header) {
// Header size
if (header.size() < 32) {
return false;
}
// Read magic and version at once
byte magicVersion[12];
if (header.read(magicVersion, sizeof(magicVersion)) != sizeof(magicVersion)) {
return false;
}
if (memcmp(magicVersion, "CRYO_APC1.20", sizeof(magicVersion))) {
return false;
}
//uint32 num_samples = header.readUint32LE();
header.skip(4);
uint32 samplerate = header.readUint32LE();
if (_sampleRate != uint32(-1) && _sampleRate != samplerate) {
error("Samplerate mismatch");
return false;
}
_sampleRate = samplerate;
uint32 leftSample = header.readSint32LE();
uint32 rightSample = header.readSint32LE();
uint32 audioFlags = header.readUint32LE();
byte stereo = (audioFlags & 1);
if (_stereo != byte(-1) && _stereo != stereo) {
error("Channels mismatch");
return false;
}
_stereo = stereo;
_status[0].last = leftSample;
_status[1].last = rightSample;
_status[0].stepIndex = _status[1].stepIndex = 0;
_status[0].step = _status[1].step = Ima_ADPCMStream::_imaTable[0];
if (!_audStream) {
_audStream.reset(makeQueuingAudioStream(_sampleRate, _stereo));
}
return true;
}
void APCStreamImpl::queuePacket(Common::SeekableReadStream *data) {
Common::ScopedPtr<Common::SeekableReadStream> packet(data);
uint32 size = packet->size() - packet->pos();
if (size == 0) {
return;
}
// From 4-bits samples to 16-bits
int16 *outputBuffer = (int16 *)malloc(size * 4);
int16 *outputPtr = outputBuffer;
int channelOffset = (_stereo ? 1 : 0);
for (uint32 counter = size; counter > 0; counter--) {
byte word = packet->readByte();
*(outputPtr++) = decodeIMA((word >> 4) & 0x0f, _status);
*(outputPtr++) = decodeIMA((word >> 0) & 0x0f, _status + channelOffset);
}
byte flags = FLAG_16BITS;
if (_stereo) {
flags |= FLAG_STEREO;
}
#ifdef SCUMM_LITTLE_ENDIAN
flags |= Audio::FLAG_LITTLE_ENDIAN;
#endif
_audStream->queueBuffer((byte *)outputBuffer, size * 4, DisposeAfterUse::YES, flags);
}
int16 APCStreamImpl::decodeIMA(byte code, Status *status) {
int32 E = (2 * (code & 0x7) + 1) * status->step / 8;
int32 diff = (code & 0x08) ? -E : E;
// In Cryo APC data is only truncated and not CLIPed as expected
int16 samp = (status->last + diff);
int32 index = status->stepIndex + Ima_ADPCMStream::_stepAdjustTable[code];
index = CLIP<int32>(index, 0, ARRAYSIZE(Ima_ADPCMStream::_imaTable) - 1);
status->last = samp;
status->stepIndex = index;
status->step = Ima_ADPCMStream::_imaTable[index];
return samp;
}
PacketizedAudioStream *makeAPCStream(Common::SeekableReadStream &header) {
Common::ScopedPtr<APCStream> stream(new APCStreamImpl());
if (!stream->init(header)) {
return nullptr;
}
return stream.release();
}
APCStream *makeAPCStream(uint sampleRate, bool stereo) {
assert((sampleRate % 11025) == 0);
return new APCStreamImpl(sampleRate, stereo ? 1 : 0);
}
} // End of namespace Audio