mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-05 09:10:29 +00:00
deb46ab38b
IMA is really just the definition on how to decode a sample from a nibble, DVI is just a way for those nibbles to be stored in the stream.
236 lines
6.7 KiB
C++
236 lines
6.7 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 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.
|
|
*
|
|
* $URL$
|
|
* $Id$
|
|
*
|
|
*/
|
|
|
|
/**
|
|
* Internal interfaces to the ADPCM encoders.
|
|
*
|
|
* These can be used to make custom ADPCM decoder subclasses,
|
|
* or to at least share some common data tables between various
|
|
* ADPCM decoder implementations.
|
|
*/
|
|
|
|
#ifndef SOUND_ADPCM_INTERN_H
|
|
#define SOUND_ADPCM_INTERN_H
|
|
|
|
#include "audio/audiostream.h"
|
|
#include "common/endian.h"
|
|
#include "common/stream.h"
|
|
|
|
|
|
namespace Audio {
|
|
|
|
class ADPCMStream : public RewindableAudioStream {
|
|
protected:
|
|
Common::SeekableReadStream *_stream;
|
|
const DisposeAfterUse::Flag _disposeAfterUse;
|
|
const int32 _startpos;
|
|
const int32 _endpos;
|
|
const int _channels;
|
|
const uint32 _blockAlign;
|
|
uint32 _blockPos[2];
|
|
const int _rate;
|
|
|
|
struct {
|
|
// OKI/IMA
|
|
struct {
|
|
int32 last;
|
|
int32 stepIndex;
|
|
} ima_ch[2];
|
|
} _status;
|
|
|
|
virtual void reset();
|
|
|
|
public:
|
|
ADPCMStream(Common::SeekableReadStream *stream, DisposeAfterUse::Flag disposeAfterUse, uint32 size, int rate, int channels, uint32 blockAlign);
|
|
~ADPCMStream();
|
|
|
|
virtual bool endOfData() const { return (_stream->eos() || _stream->pos() >= _endpos); }
|
|
virtual bool isStereo() const { return _channels == 2; }
|
|
virtual int getRate() const { return _rate; }
|
|
|
|
virtual bool rewind();
|
|
|
|
|
|
/**
|
|
* This table is used by some ADPCM variants (IMA and OKI) to adjust the
|
|
* step for use on the next sample.
|
|
* The first 8 entries are identical to the second 8 entries. Hence, we
|
|
* could half the table in size. But since the lookup index is always a
|
|
* 4-bit nibble, it is more efficient to just keep it as it is.
|
|
*/
|
|
static const int16 _stepAdjustTable[16];
|
|
};
|
|
|
|
class Oki_ADPCMStream : public ADPCMStream {
|
|
public:
|
|
Oki_ADPCMStream(Common::SeekableReadStream *stream, DisposeAfterUse::Flag disposeAfterUse, uint32 size, int rate, int channels, uint32 blockAlign)
|
|
: ADPCMStream(stream, disposeAfterUse, size, rate, channels, blockAlign) {}
|
|
|
|
virtual int readBuffer(int16 *buffer, const int numSamples);
|
|
|
|
protected:
|
|
int16 decodeOKI(byte);
|
|
};
|
|
|
|
class Ima_ADPCMStream : public ADPCMStream {
|
|
protected:
|
|
int16 decodeIMA(byte code, int channel = 0); // Default to using the left channel/using one channel
|
|
|
|
public:
|
|
Ima_ADPCMStream(Common::SeekableReadStream *stream, DisposeAfterUse::Flag disposeAfterUse, uint32 size, int rate, int channels, uint32 blockAlign)
|
|
: ADPCMStream(stream, disposeAfterUse, size, rate, channels, blockAlign) {
|
|
memset(&_status, 0, sizeof(_status));
|
|
}
|
|
|
|
/**
|
|
* This table is used by decodeIMA.
|
|
*/
|
|
static const int16 _imaTable[89];
|
|
};
|
|
|
|
class DVI_ADPCMStream : public Ima_ADPCMStream {
|
|
public:
|
|
DVI_ADPCMStream(Common::SeekableReadStream *stream, DisposeAfterUse::Flag disposeAfterUse, uint32 size, int rate, int channels, uint32 blockAlign)
|
|
: Ima_ADPCMStream(stream, disposeAfterUse, size, rate, channels, blockAlign) {}
|
|
|
|
virtual int readBuffer(int16 *buffer, const int numSamples);
|
|
};
|
|
|
|
class Apple_ADPCMStream : public Ima_ADPCMStream {
|
|
protected:
|
|
// Apple QuickTime IMA ADPCM
|
|
int32 _streamPos[2];
|
|
int16 _buffer[2][2];
|
|
uint8 _chunkPos[2];
|
|
|
|
void reset() {
|
|
Ima_ADPCMStream::reset();
|
|
_chunkPos[0] = 0;
|
|
_chunkPos[1] = 0;
|
|
_streamPos[0] = 0;
|
|
_streamPos[1] = _blockAlign;
|
|
}
|
|
|
|
public:
|
|
Apple_ADPCMStream(Common::SeekableReadStream *stream, DisposeAfterUse::Flag disposeAfterUse, uint32 size, int rate, int channels, uint32 blockAlign)
|
|
: Ima_ADPCMStream(stream, disposeAfterUse, size, rate, channels, blockAlign) {
|
|
_chunkPos[0] = 0;
|
|
_chunkPos[1] = 0;
|
|
_streamPos[0] = 0;
|
|
_streamPos[1] = _blockAlign;
|
|
}
|
|
|
|
virtual int readBuffer(int16 *buffer, const int numSamples);
|
|
|
|
};
|
|
|
|
class MSIma_ADPCMStream : public Ima_ADPCMStream {
|
|
public:
|
|
MSIma_ADPCMStream(Common::SeekableReadStream *stream, DisposeAfterUse::Flag disposeAfterUse, uint32 size, int rate, int channels, uint32 blockAlign, bool invertSamples = false)
|
|
: Ima_ADPCMStream(stream, disposeAfterUse, size, rate, channels, blockAlign), _invertSamples(invertSamples) {
|
|
if (blockAlign == 0)
|
|
error("ADPCMStream(): blockAlign isn't specified for MS IMA ADPCM");
|
|
}
|
|
|
|
virtual int readBuffer(int16 *buffer, const int numSamples) {
|
|
if (_channels == 1)
|
|
return readBufferMSIMA1(buffer, numSamples);
|
|
else
|
|
return readBufferMSIMA2(buffer, numSamples);
|
|
}
|
|
|
|
int readBufferMSIMA1(int16 *buffer, const int numSamples);
|
|
int readBufferMSIMA2(int16 *buffer, const int numSamples);
|
|
|
|
private:
|
|
bool _invertSamples; // Some implementations invert the way samples are decoded
|
|
};
|
|
|
|
class MS_ADPCMStream : public ADPCMStream {
|
|
protected:
|
|
struct ADPCMChannelStatus {
|
|
byte predictor;
|
|
int16 delta;
|
|
int16 coeff1;
|
|
int16 coeff2;
|
|
int16 sample1;
|
|
int16 sample2;
|
|
};
|
|
|
|
struct {
|
|
// MS ADPCM
|
|
ADPCMChannelStatus ch[2];
|
|
} _status;
|
|
|
|
void reset() {
|
|
ADPCMStream::reset();
|
|
memset(&_status, 0, sizeof(_status));
|
|
}
|
|
|
|
public:
|
|
MS_ADPCMStream(Common::SeekableReadStream *stream, DisposeAfterUse::Flag disposeAfterUse, uint32 size, int rate, int channels, uint32 blockAlign)
|
|
: ADPCMStream(stream, disposeAfterUse, size, rate, channels, blockAlign) {
|
|
if (blockAlign == 0)
|
|
error("MS_ADPCMStream(): blockAlign isn't specified for MS ADPCM");
|
|
memset(&_status, 0, sizeof(_status));
|
|
}
|
|
|
|
virtual int readBuffer(int16 *buffer, const int numSamples);
|
|
|
|
protected:
|
|
int16 decodeMS(ADPCMChannelStatus *c, byte);
|
|
};
|
|
|
|
// Duck DK3 IMA ADPCM Decoder
|
|
// Based on FFmpeg's decoder and http://wiki.multimedia.cx/index.php?title=Duck_DK3_IMA_ADPCM
|
|
|
|
class DK3_ADPCMStream : public Ima_ADPCMStream {
|
|
protected:
|
|
|
|
void reset() {
|
|
Ima_ADPCMStream::reset();
|
|
_topNibble = false;
|
|
}
|
|
|
|
public:
|
|
DK3_ADPCMStream(Common::SeekableReadStream *stream, DisposeAfterUse::Flag disposeAfterUse, uint32 size, int rate, int channels, uint32 blockAlign)
|
|
: Ima_ADPCMStream(stream, disposeAfterUse, size, rate, channels, blockAlign) {
|
|
|
|
// DK3 only works as a stereo stream
|
|
assert(channels == 2);
|
|
_topNibble = false;
|
|
}
|
|
|
|
virtual int readBuffer(int16 *buffer, const int numSamples);
|
|
|
|
private:
|
|
byte _nibble, _lastByte;
|
|
bool _topNibble;
|
|
};
|
|
|
|
} // End of namespace Audio
|
|
|
|
#endif
|