scummvm/scumm/smush/imuse_channel.cpp

328 lines
9 KiB
C++
Raw Normal View History

2002-08-24 15:31:37 +00:00
/* ScummVM - Scumm Interpreter
* Copyright (C) 2001/2002 The ScummVM project
*
* 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*
* $Header$
*
*/
#include <stdafx.h>
#include "channel.h"
#include "chunk.h"
#include "chunk_type.h"
2002-08-24 15:31:37 +00:00
#include <assert.h>
2002-08-30 07:24:45 +00:00
#include <string.h>
2002-08-24 15:31:37 +00:00
2002-08-30 07:24:45 +00:00
ImuseChannel::ImuseChannel(int32 track, int32 freq) :
2002-08-24 15:31:37 +00:00
_track(track),
_tbuffer(0),
_tbufferSize(0),
_sbuffer(0),
_sbufferSize(0),
_frequency(freq),
_dataSize(-1),
_inData(false) {
}
ImuseChannel::~ImuseChannel() {
if(_tbuffer) {
delete []_tbuffer;
}
if(_sbuffer) {
warning("_sbuffer should be 0 !!!");
delete []_sbuffer;
}
}
bool ImuseChannel::isTerminated() const {
return (_dataSize <= 0 && _sbuffer == 0);
}
2002-08-30 07:24:45 +00:00
bool ImuseChannel::setParameters(int32 nbframes, int32 size, int32 unk1, int32 unk2) {
2002-08-24 15:31:37 +00:00
return true;
}
2002-08-30 07:24:45 +00:00
bool ImuseChannel::checkParameters(int32 index, int32 nbframes, int32 size, int32 unk1, int32 unk2) {
2002-08-24 15:31:37 +00:00
return true;
}
2002-08-30 07:24:45 +00:00
bool ImuseChannel::appendData(Chunk & b, int32 size) {
2002-08-24 15:31:37 +00:00
if(_dataSize == -1) { // First call
assert(size > 8);
2002-08-24 23:22:10 +00:00
Chunk::type imus_type = b.getDword(); imus_type = SWAP_BYTES(imus_type);
2002-08-30 07:24:45 +00:00
uint32 imus_size = b.getDword(); imus_size = SWAP_BYTES(imus_size);
if(imus_type != TYPE_iMUS) error("Invalid Chunk for imuse_channel");
2002-08-24 15:31:37 +00:00
size -= 8;
_tbufferSize = size;
assert(_tbufferSize);
2002-08-30 07:24:45 +00:00
_tbuffer = new byte[_tbufferSize];
2002-08-24 15:31:37 +00:00
if(!_tbuffer) error("imuse_channel failed to allocate memory");
b.read(_tbuffer, size);
_dataSize = -2; // even if _in_data does not get set, this won't be called again
} else {
if(_tbuffer) { // remaining from last call
2002-08-30 07:24:45 +00:00
byte * old = _tbuffer;
int32 new_size = size + _tbufferSize;
_tbuffer = new byte[new_size];
2002-08-24 15:31:37 +00:00
if(!_tbuffer) error("imuse_channel failed to allocate memory");
memcpy(_tbuffer, old, _tbufferSize);
delete []old;
b.read(_tbuffer + _tbufferSize, size);
_tbufferSize += size;
} else {
_tbufferSize = size;
2002-08-30 07:24:45 +00:00
_tbuffer = new byte[_tbufferSize];
2002-08-24 15:31:37 +00:00
if(!_tbuffer) error("imuse_channel failed to allocate memory");
b.read(_tbuffer, size);
}
}
return processBuffer();
}
bool ImuseChannel::handleFormat(Chunk & src) {
if(src.getSize() != 20) error("invalid size for FRMT Chunk");
2002-08-30 07:24:45 +00:00
uint32 imuse_start = src.getDword();
2002-08-24 23:22:10 +00:00
imuse_start = SWAP_BYTES(imuse_start);
2002-08-24 15:31:37 +00:00
src.seek(4);
_bitsize = src.getDword();
2002-08-24 23:22:10 +00:00
_bitsize = SWAP_BYTES(_bitsize);
2002-08-24 15:31:37 +00:00
_rate = src.getDword();
2002-08-24 23:22:10 +00:00
_rate = SWAP_BYTES(_rate);
2002-08-24 15:31:37 +00:00
_channels = src.getDword();
2002-08-24 23:22:10 +00:00
_channels = SWAP_BYTES(_channels);
2002-08-24 15:31:37 +00:00
assert(_channels == 1 || _channels == 2);
return true;
}
bool ImuseChannel::handleText(Chunk & src) {
2002-08-24 15:31:37 +00:00
return true;
}
bool ImuseChannel::handleRegion(Chunk & src) {
if(src.getSize() != 8) error("invalid size for REGN Chunk");
2002-08-24 15:31:37 +00:00
return true;
}
bool ImuseChannel::handleStop(Chunk & src) {
if(src.getSize() != 4) error("invalid size for STOP Chunk");
2002-08-24 15:31:37 +00:00
return true;
}
bool ImuseChannel::handleMap(Chunk & map) {
2002-08-24 15:31:37 +00:00
while(!map.eof()) {
Chunk * sub = map.subBlock();
2002-08-24 15:31:37 +00:00
switch(sub->getType()) {
case TYPE_FRMT:
handleFormat(*sub);
break;
case TYPE_TEXT:
handleText(*sub);
break;
case TYPE_REGN:
handleRegion(*sub);
break;
case TYPE_STOP:
handleStop(*sub);
break;
default:
error("Unknown iMUS subChunk found : %s, %d", Chunk::ChunkString(sub->getType()), sub->getSize());
2002-08-24 15:31:37 +00:00
}
delete sub;
}
return true;
}
void ImuseChannel::decode() {
int remaining_size = _sbufferSize % 3;
if(remaining_size) {
_srbufferSize -= remaining_size;
assert(_inData);
if(_tbuffer == 0) {
2002-08-30 07:24:45 +00:00
_tbuffer = new byte[remaining_size];
2002-08-24 15:31:37 +00:00
memcpy(_tbuffer, _sbuffer + _sbufferSize - remaining_size, remaining_size);
_tbufferSize = remaining_size;
_sbufferSize -= remaining_size;
} else {
warning("impossible ! : %p, %d, %d, %p(%d), %p(%d, %d)",
this, _dataSize, _inData, _tbuffer, _tbufferSize, _sbuffer, _sbufferSize, _srbufferSize);
2002-08-30 07:24:45 +00:00
byte * old = _tbuffer;
2002-08-24 15:31:37 +00:00
int new_size = remaining_size + _tbufferSize;
2002-08-30 07:24:45 +00:00
_tbuffer = new byte[new_size];
2002-08-24 15:31:37 +00:00
if(!_tbuffer) error("imuse_channel failed to allocate memory");
memcpy(_tbuffer, old, _tbufferSize);
delete []old;
memcpy(_tbuffer + _tbufferSize, _sbuffer + _sbufferSize - remaining_size, remaining_size);
_tbufferSize += remaining_size;
}
}
int loop_size = _sbufferSize / 3;
int new_size = loop_size * 2;
short * keep, * decoded;
2002-08-30 07:24:45 +00:00
keep = decoded = new int16[new_size];
2002-08-24 15:31:37 +00:00
assert(keep);
unsigned char * source = _sbuffer;
while(loop_size--) {
int v1 = *source++;
int v2 = *source++;
int v3 = *source++;
int value = (((v2 & 0x0f) << 12) | (v1 << 4)) - 0x8000;
2002-08-30 07:24:45 +00:00
*decoded++ = (int16)value;
2002-08-24 15:31:37 +00:00
value = (((v2 & 0xf0) << 8) | (v3 << 4)) - 0x8000;
2002-08-30 07:24:45 +00:00
*decoded++ = (int16)value;
2002-08-24 15:31:37 +00:00
}
delete []_sbuffer;
2002-08-30 07:24:45 +00:00
_sbuffer = (byte *)keep;
_sbufferSize = new_size * sizeof(int16);
2002-08-24 15:31:37 +00:00
}
2002-08-30 07:24:45 +00:00
bool ImuseChannel::handleSubTags(int32 & offset) {
2002-08-24 21:57:00 +00:00
if(_tbufferSize - offset >= 8) {
Chunk::type type = READ_BE_UINT32(_tbuffer + offset);
2002-08-30 07:24:45 +00:00
uint32 size = READ_BE_UINT32(_tbuffer + offset + 4);
uint32 available_size = _tbufferSize - offset;
2002-08-24 15:31:37 +00:00
switch(type) {
case TYPE_MAP_:
_inData = false;
if(available_size >= (size + 8)) {
2002-08-30 07:24:45 +00:00
ContChunk c((byte *)_tbuffer + offset);
2002-08-24 15:31:37 +00:00
handleMap(c);
}
break;
case TYPE_DATA: // Sound data !!!
_inData = true;
_dataSize = size;
offset += 8;
{
int reqsize = 1;
2002-08-30 07:24:45 +00:00
if(_channels == 2)
reqsize *= 2;
if(_bitsize == 16)
reqsize *= 2;
2002-08-24 15:31:37 +00:00
else if(_bitsize == 12) {
if(reqsize > 1)
reqsize = reqsize * 3 / 2;
else reqsize = 3;
}
if((size % reqsize) != 0) {
warning("Invalid iMUS sound data size : (%d %% %d) != 0, correcting...", size, reqsize);
size += 3 - (size % reqsize);
}
}
return false;
default:
error("unknown Chunk in iMUS track : %s ", Chunk::ChunkString(type));
2002-08-24 15:31:37 +00:00
}
offset += size + 8;
return true;
}
return false;
}
bool ImuseChannel::processBuffer() {
// see comments in saud_channel::processBuffer for an explanation
assert(_tbuffer != 0);
assert(_tbufferSize != 0);
assert(_sbuffer == 0);
assert(_sbufferSize == 0);
if(_inData) {
if(_dataSize < _tbufferSize) {
2002-08-30 07:24:45 +00:00
int32 offset= _dataSize;
2002-08-24 15:31:37 +00:00
while(handleSubTags(offset));
_sbufferSize = _dataSize;
_sbuffer = _tbuffer;
if(offset < _tbufferSize) { // there is still some unprocessed data
2002-08-30 07:24:45 +00:00
int32 new_size = _tbufferSize - offset;
_tbuffer = new byte[new_size];
if(!_tbuffer) error("imuse_channel failed to allocate memory");
2002-08-24 15:31:37 +00:00
memcpy(_tbuffer, _sbuffer + offset, new_size);
_tbufferSize = new_size;
} else {
_tbuffer = 0;
_tbufferSize = 0;
}
if(_sbufferSize == 0) {
// this never happened yet, but who knows
delete []_sbuffer;
_sbuffer = 0;
}
} else {
// easy, swap the buffer
_sbufferSize = _tbufferSize;
_sbuffer = _tbuffer;
_tbufferSize = 0;
_tbuffer = 0;
}
} else {
2002-08-30 07:24:45 +00:00
int32 offset = 0;
2002-08-24 15:31:37 +00:00
while(handleSubTags(offset));
if(_inData) {
_sbufferSize = _tbufferSize - offset;
assert(_sbufferSize);
2002-08-30 07:24:45 +00:00
_sbuffer = new byte[_sbufferSize];
if(!_sbuffer) error("imuse_channel failed to allocate memory");
2002-08-24 15:31:37 +00:00
memcpy(_sbuffer, _tbuffer + offset, _sbufferSize);
delete []_tbuffer;
_tbuffer = 0;
_tbufferSize = 0;
} else {
if(offset) { // maybe I should assert() this to avoid a lock...
2002-08-30 07:24:45 +00:00
byte * old = _tbuffer;
int32 new_size = _tbufferSize - offset;
_tbuffer = new byte[new_size];
if(!_tbuffer) error("imuse_channel failed to allocate memory");
2002-08-24 15:31:37 +00:00
memcpy(_tbuffer, old + offset, new_size);
_tbufferSize = new_size;
delete []old;
}
}
}
_srbufferSize = _sbufferSize;
if(_sbuffer && _bitsize == 12) decode();
return true;
}
2002-08-30 07:24:45 +00:00
int32 ImuseChannel::availableSoundData(void) const {
int32 ret = _sbufferSize;
2002-08-24 15:31:37 +00:00
if(_channels == 2) ret /= 2;
if(_bitsize > 8) ret /= 2;
return ret;
}
2002-08-30 07:24:45 +00:00
void ImuseChannel::getSoundData(int16 * snd, int32 size) {
2002-08-24 15:31:37 +00:00
if(_dataSize <= 0 || _bitsize <= 8) error("invalid call to imuse_channel::read_sound_data()");
if(_channels == 2) size *= 2;
2002-08-30 07:24:45 +00:00
for(int32 i = 0; i < size; i++)
2002-08-24 15:31:37 +00:00
snd[i] = READ_BE_UINT16(_sbuffer + 2 * i);
delete []_sbuffer;
assert(_sbufferSize == 2 * size);
_sbuffer = 0;
_sbufferSize = 0;
_dataSize -= _srbufferSize;
}
2002-08-30 07:24:45 +00:00
void ImuseChannel::getSoundData(int8 * snd, int32 size) {
2002-08-24 15:31:37 +00:00
if(_dataSize <= 0 || _bitsize > 8) error("invalid call to imuse_channel::read_sound_data()");
if(_channels == 2) size *= 2;
2002-08-30 07:24:45 +00:00
for(int32 i = 0; i < size; i++)
2002-08-24 15:31:37 +00:00
snd[i] = _sbuffer[i];
delete []_sbuffer;
_sbuffer = 0;
_sbufferSize = 0;
_dataSize -= _srbufferSize;
}