2006-11-06 13:40:24 +00:00
|
|
|
/* ScummVM - Scumm Interpreter
|
|
|
|
* Copyright (C) 2006 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*
|
|
|
|
* $URL$
|
|
|
|
* $Id$
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "common/stdafx.h"
|
|
|
|
#include "common/system.h"
|
|
|
|
#include "scumm/scumm.h"
|
|
|
|
#include "scumm/he/intern_he.h"
|
|
|
|
#include "scumm/he/cup_player_he.h"
|
|
|
|
|
|
|
|
namespace Scumm {
|
|
|
|
|
|
|
|
bool CUP_Player::open(const char *filename) {
|
|
|
|
bool opened = false;
|
|
|
|
debug(1, "opening '%s'", filename);
|
|
|
|
if (_fd.open(filename)) {
|
|
|
|
uint32 tag = _fd.readUint32BE();
|
|
|
|
_fd.readUint32BE();
|
|
|
|
if (tag == 'BEAN') {
|
|
|
|
_playbackRate = 66;
|
|
|
|
_width = 640;
|
|
|
|
_height = 480;
|
|
|
|
parseHeaderTags();
|
|
|
|
debug(1, "rate %d width %d height %d", _playbackRate, _width, _height);
|
|
|
|
_offscreenBuffer = (uint8 *)malloc(_width * _height);
|
|
|
|
memset(_offscreenBuffer, 0, _width * _height);
|
|
|
|
opened = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return opened;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CUP_Player::close() {
|
|
|
|
_fd.close();
|
|
|
|
free(_offscreenBuffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32 CUP_Player::loadNextChunk() {
|
|
|
|
uint32 tag = _fd.readUint32BE();
|
|
|
|
uint32 size = _fd.readUint32BE() - 8;
|
|
|
|
if (_currentChunkSize < size) {
|
|
|
|
free(_currentChunkData);
|
|
|
|
_currentChunkSize = 0;
|
|
|
|
_currentChunkData = (uint8 *)malloc(size);
|
|
|
|
}
|
|
|
|
if (_currentChunkData) {
|
|
|
|
_currentChunkSize = size;
|
|
|
|
_fd.read(_currentChunkData, _currentChunkSize);
|
|
|
|
}
|
|
|
|
return tag;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CUP_Player::parseHeaderTags() {
|
|
|
|
_dataSize = 0;
|
|
|
|
while (_dataSize == 0 && !_vm->_quit && !_fd.ioFailed()) {
|
|
|
|
uint32 tag = loadNextChunk();
|
|
|
|
switch (tag) {
|
|
|
|
case 'HEAD':
|
|
|
|
handleHEAD(_currentChunkData, _currentChunkSize);
|
|
|
|
break;
|
|
|
|
case 'SFXB':
|
|
|
|
handleSFXB(_currentChunkData, _currentChunkSize);
|
|
|
|
break;
|
|
|
|
case 'RGBS':
|
|
|
|
handleRGBS(_currentChunkData, _currentChunkSize);
|
|
|
|
break;
|
|
|
|
case 'DATA':
|
|
|
|
_dataSize = _currentChunkSize;
|
|
|
|
break;
|
|
|
|
case 'GFXB':
|
|
|
|
// this is never triggered
|
|
|
|
default:
|
|
|
|
warning("unhandled tag %c%c%c%c", tag>>24, (tag>>16)&0xFF, (tag>>8)&0xFF, tag&0xFF);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CUP_Player::play() {
|
|
|
|
int ticks = _system->getMillis();
|
|
|
|
Common::Rect r;
|
|
|
|
while (_currentChunkSize != 0 && !_vm->_quit) {
|
|
|
|
uint32 tag, size;
|
|
|
|
parseNextTag(_currentChunkData, &r, tag, size);
|
|
|
|
if (tag == 'BLOK') {
|
|
|
|
bool fastMode = false;
|
|
|
|
int diff = _system->getMillis() - ticks;
|
|
|
|
if (diff >= 0 && diff <= _playbackRate && !fastMode) {
|
|
|
|
_system->delayMillis(_playbackRate - diff);
|
|
|
|
} else {
|
|
|
|
_system->delayMillis(1);
|
|
|
|
}
|
|
|
|
_system->setPalette(_paletteData, 0, 256);
|
|
|
|
_system->copyRectToScreen(_offscreenBuffer, _width, 0, 0, _width, _height);
|
|
|
|
_system->updateScreen();
|
|
|
|
_vm->parseEvents();
|
|
|
|
|
|
|
|
ticks = _system->getMillis();
|
|
|
|
size = 8;
|
|
|
|
}
|
|
|
|
_currentChunkData += size;
|
|
|
|
_currentChunkSize -= size;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CUP_Player::parseNextTag(const uint8 *data, Common::Rect *r1, uint32 &tag, uint32 &size) {
|
|
|
|
tag = READ_BE_UINT32(data);
|
|
|
|
size = READ_BE_UINT32(data + 4);
|
|
|
|
data += 8;
|
|
|
|
switch (tag) {
|
|
|
|
case 'FRAM':
|
|
|
|
handleFRAM(_offscreenBuffer, data, size);
|
|
|
|
break;
|
|
|
|
case 'LZSS':
|
|
|
|
data = handleLZSS(data, size);
|
|
|
|
if (data) {
|
|
|
|
uint32 t, s;
|
|
|
|
parseNextTag(data, r1, t, s);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'RATE':
|
|
|
|
handleRATE(data, size);
|
|
|
|
break;
|
|
|
|
case 'RGBS':
|
|
|
|
handleRGBS(data, size);
|
|
|
|
break;
|
|
|
|
case 'SNDE':
|
|
|
|
handleSNDE(data, size);
|
|
|
|
break;
|
|
|
|
case 'TOIL':
|
|
|
|
handleTOIL(data, size);
|
|
|
|
break;
|
|
|
|
case 'BLOK':
|
|
|
|
// not handled here
|
|
|
|
break;
|
|
|
|
case 'SRLE':
|
|
|
|
case 'WRLE':
|
|
|
|
// these are never triggered
|
|
|
|
default:
|
|
|
|
warning("Unhandled tag %c%c%c%c", tag>>24, (tag>>16)&0xFF, (tag>>8)&0xFF, tag&0xFF);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CUP_Player::handleHEAD(const uint8 *data, uint32 dataSize) {
|
|
|
|
_playbackRate = READ_LE_UINT16(data);
|
|
|
|
_width = READ_LE_UINT16(data + 2);
|
|
|
|
_height = READ_LE_UINT16(data + 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CUP_Player::handleSFXB(const uint8 *data, uint32 dataSize) {
|
|
|
|
// TODO
|
|
|
|
warning("unhandled SFXB");
|
|
|
|
}
|
|
|
|
|
|
|
|
void CUP_Player::handleRGBS(const uint8 *data, uint32 dataSize) {
|
|
|
|
int i;
|
|
|
|
uint8 *ptr = _paletteData;
|
|
|
|
|
|
|
|
for (i = 0; i < 256; i++) {
|
|
|
|
*ptr++ = *data++;
|
|
|
|
*ptr++ = *data++;
|
|
|
|
*ptr++ = *data++;
|
|
|
|
*ptr++ = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CUP_Player::handleFRAM(uint8 *dst, const uint8 *data, uint32 size) {
|
|
|
|
Common::Rect r1;
|
|
|
|
memset(&r1, 0, sizeof(r1));
|
|
|
|
int code = 256;
|
|
|
|
int flags = *data++;
|
|
|
|
if (flags & 1) {
|
|
|
|
code = *data++;
|
|
|
|
}
|
|
|
|
if (flags & 2) {
|
|
|
|
r1.left = READ_LE_UINT16(data); data += 2;
|
|
|
|
r1.top = READ_LE_UINT16(data); data += 2;
|
|
|
|
r1.right = READ_LE_UINT16(data); data += 2;
|
|
|
|
r1.bottom = READ_LE_UINT16(data); data += 2;
|
|
|
|
}
|
|
|
|
if (flags & 0x80) {
|
|
|
|
decodeFRAM(dst, &r1, data, code);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// decodeWiz, decodeTRLE
|
|
|
|
void CUP_Player::decodeFRAM(uint8 *dst, Common::Rect *r1, const uint8 *data, int code1) {
|
|
|
|
if (code1 == 256) {
|
|
|
|
dst += r1->top * _width + r1->left;
|
|
|
|
int h = r1->bottom - r1->top + 1;
|
|
|
|
int w = r1->right - r1->left + 1;
|
|
|
|
if (h < 0 || w < 0) { warning("h=%d w=%d", h, w); return; }
|
|
|
|
while (h--) {
|
|
|
|
uint16 lineSize = READ_LE_UINT16(data); data += 2;
|
|
|
|
uint8 *dstNextLine = dst + _width;
|
|
|
|
const uint8 *dataNextLine = data + lineSize;
|
|
|
|
if (lineSize != 0) {
|
|
|
|
uint8 *dstEnd = dst + w;
|
|
|
|
while (dst < dstEnd) {
|
|
|
|
int code = *data++;
|
|
|
|
if (code & 1) { // skip
|
|
|
|
code >>= 1;
|
|
|
|
dst += code;
|
|
|
|
} else if (code & 2) { // set
|
|
|
|
code = (code >> 2) + 1;
|
2006-11-06 18:21:07 +00:00
|
|
|
const int sz = MIN((int)code, (int)(dstEnd - dst));
|
2006-11-06 13:40:24 +00:00
|
|
|
memset(dst, *data++, sz);
|
|
|
|
dst += sz;
|
|
|
|
} else { // copy
|
|
|
|
code = (code >> 2) + 1;
|
2006-11-06 18:21:07 +00:00
|
|
|
const int sz = MIN((int)code, (int)(dstEnd - dst));
|
2006-11-06 13:40:24 +00:00
|
|
|
memcpy(dst, data, sz);
|
|
|
|
dst += sz;
|
|
|
|
data += sz;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
dst = dstNextLine;
|
|
|
|
data = dataNextLine;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8 *CUP_Player::handleLZSS(const uint8 *data, uint32 dataSize) {
|
|
|
|
uint32 compressionType = 0;
|
|
|
|
uint32 compressionSize = 0;
|
|
|
|
uint32 tag, size;
|
|
|
|
|
|
|
|
tag = READ_BE_UINT32(data);
|
|
|
|
size = READ_BE_UINT32(data + 4);
|
|
|
|
if (tag == 'LZHD') {
|
|
|
|
compressionType = READ_LE_UINT32(data + 8);
|
|
|
|
compressionSize = READ_LE_UINT32(data + 12);
|
|
|
|
}
|
|
|
|
data += size;
|
|
|
|
|
|
|
|
tag = READ_BE_UINT32(data);
|
|
|
|
size = READ_BE_UINT32(data + 4);
|
|
|
|
if (tag == 'DATA') {
|
|
|
|
if (compressionType == 0x2000) {
|
|
|
|
if (_bufferLzssSize < compressionSize) {
|
|
|
|
_bufferLzssSize = compressionSize;
|
|
|
|
free(_bufferLzssData);
|
|
|
|
_bufferLzssData = (uint8 *)malloc(_bufferLzssSize);
|
|
|
|
}
|
|
|
|
data += 8;
|
|
|
|
uint32 offset1 = READ_LE_UINT32(data);
|
|
|
|
uint32 offset2 = READ_LE_UINT32(data + 4);
|
|
|
|
decodeLzssData(_bufferLzssData, data + 8, data + offset1, data + offset2, _tempLzssBuffer);
|
|
|
|
return _bufferLzssData;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CUP_Player::decodeLzssData(uint8 *dst1, const uint8 *src1, const uint8 *src2, const uint8 *src3, uint8 *dst2) {
|
|
|
|
int index = 1;
|
|
|
|
while (1) {
|
|
|
|
int code = *src1++;
|
|
|
|
for (int b = 0; b < 8; ++b) {
|
|
|
|
if (code & (1 << b)) {
|
|
|
|
*dst1++ = dst2[index] = *src2++;
|
|
|
|
++index;
|
|
|
|
index &= 0xFFF;
|
|
|
|
} else {
|
|
|
|
int cmd = READ_LE_UINT16(src3); src3 += 2;
|
|
|
|
int count = (cmd >> 0xC) + 2;
|
|
|
|
int offs = cmd & 0xFFF;
|
|
|
|
if (offs == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
while (count--) {
|
|
|
|
*dst1++ = dst2[index] = dst2[offs];
|
|
|
|
++index;
|
|
|
|
index &= 0xFFF;
|
|
|
|
++offs;
|
|
|
|
offs &= 0xFFF;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CUP_Player::handleRATE(const uint8 *data, uint32 dataSize) {
|
|
|
|
_playbackRate = (int16)READ_LE_UINT16(data);
|
|
|
|
if (_playbackRate > 4000) {
|
|
|
|
_playbackRate = 4000;
|
|
|
|
} else if (_playbackRate < 1) {
|
|
|
|
_playbackRate = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CUP_Player::handleSNDE(const uint8 *data, uint32 dataSize) {
|
|
|
|
// TODO
|
|
|
|
warning("unhandled SNDE");
|
|
|
|
// READ_LE_UINT32
|
|
|
|
// READ_LE_UINT16
|
|
|
|
// READ_LE_UINT16
|
|
|
|
// READ_LE_UINT16
|
|
|
|
}
|
|
|
|
|
|
|
|
void CUP_Player::handleTOIL(const uint8 *data, uint32 dataSize) {
|
|
|
|
int codesCount = READ_LE_UINT16(data); data += 2;
|
|
|
|
if (dataSize > 0) {
|
|
|
|
while (codesCount != 0) {
|
|
|
|
int codeSize = *data++;
|
|
|
|
if (codeSize == 0) {
|
|
|
|
codeSize = READ_LE_UINT16(data); data += 2;
|
|
|
|
}
|
|
|
|
int code = *data++;
|
|
|
|
if (code == 0) {
|
|
|
|
code = READ_LE_UINT16(data); data += 2;
|
|
|
|
}
|
|
|
|
switch (code) {
|
|
|
|
case 1:
|
|
|
|
_vm->_quit = true;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
// display copyright/information messagebox
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
// READ_LE_UINT32(data);
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
// restart playback
|
|
|
|
break;
|
|
|
|
case 5:
|
|
|
|
// disable normal screen update
|
|
|
|
break;
|
|
|
|
case 6:
|
|
|
|
// perform offscreen buffers swapping
|
|
|
|
break;
|
|
|
|
case 7:
|
|
|
|
// pause playback at specific frame
|
|
|
|
// READ_LE_UINT32(data);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
warning("Unhandled TOIL code=%d", code);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
dataSize -= codeSize;
|
|
|
|
data += codeSize;
|
|
|
|
--codesCount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
void CUP_Player::processSRLE(uint8 *dst1, uint8 *dst2, const uint8 *codeTable, uint8 *data, int32 unpackedSize) {
|
|
|
|
bool bit;
|
|
|
|
uint16 code;
|
|
|
|
while (unpackedSize > 0) {
|
|
|
|
code = *data++;
|
|
|
|
bit = (code & 1) == 1;
|
|
|
|
code >>= 1;
|
|
|
|
if (!bit) {
|
|
|
|
bit = (code & 1) == 1;
|
|
|
|
code >>= 1;
|
|
|
|
if (!bit) {
|
|
|
|
int size = code + 1;
|
|
|
|
unpackedSize -= size;
|
|
|
|
memcpy(dst1, dst2, size);
|
|
|
|
dst1 += size;
|
|
|
|
dst2 += size;
|
|
|
|
} else {
|
|
|
|
bit = (code & 1) == 1;
|
|
|
|
code >>= 1;
|
|
|
|
if (!bit) {
|
|
|
|
++dst2;
|
|
|
|
*dst1++ = codeTable[code];
|
|
|
|
--unpackedSize;
|
|
|
|
} else if (code == 0) {
|
|
|
|
size = 1 + *data++;
|
|
|
|
color = *data++;
|
|
|
|
color = _cup_colorRemapTable[color];
|
|
|
|
dst2 += size;
|
|
|
|
unpackedSize -= size;
|
|
|
|
memset(dst1, color, size);
|
|
|
|
dst1 += size;
|
|
|
|
} else {
|
|
|
|
size = code;
|
|
|
|
code = *data++;
|
|
|
|
code = _cup_colorRemapTable[code];
|
|
|
|
dst2 += size;
|
|
|
|
unpackedSize -= size;
|
|
|
|
memset(dst1, code, size);
|
|
|
|
dst1 += size;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (code == 0) {
|
|
|
|
code = READ_LE_UINT16(data); data += 2;
|
|
|
|
++code;
|
|
|
|
dst1 += code;
|
|
|
|
dst2 += code;
|
|
|
|
unpackedSize -= code;
|
|
|
|
} else {
|
|
|
|
dst1 += code;
|
|
|
|
dst2 += code;
|
|
|
|
unpackedSize -= code;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CUP_Player::processWRLE(uint8 *dst, uint8 *codeTable, const uint8 *data, int32 unpackedSize, uint8 rleCode, int pitch) {
|
|
|
|
uint8 color;
|
|
|
|
uint16 code;
|
|
|
|
if (unpackedSize <= 0) return;
|
|
|
|
do {
|
|
|
|
code = *data++;
|
|
|
|
if ((code & 1) != 0) {
|
|
|
|
int size = ((code & 6) >> 1) + 3;
|
|
|
|
code >>= 3;
|
|
|
|
unpackedSize -= var4;
|
|
|
|
color = codeTable[code];
|
|
|
|
if (color != rleCode) {
|
|
|
|
memset(dst, color, size);
|
|
|
|
} else {
|
|
|
|
memcpy(dst, dst + pitch, size);
|
|
|
|
}
|
|
|
|
} else if ((code & 2) == 0) {
|
|
|
|
code >>= 2;
|
|
|
|
if (code == 0) {
|
|
|
|
code = READ_LE_UINT16(data); data += 2;
|
|
|
|
}
|
|
|
|
dst += code;
|
|
|
|
unpackedSize -= code;
|
|
|
|
} else if ((code & 4) != 0) {
|
|
|
|
code >>= 3;
|
|
|
|
if (code == 0) {
|
|
|
|
code = *data++;
|
|
|
|
}
|
|
|
|
unpackedSize -= code;
|
|
|
|
color = *data++;
|
|
|
|
if (color != rleCode) {
|
|
|
|
memset(dst, color, code);
|
|
|
|
} else {
|
|
|
|
assert(code < pitch);
|
|
|
|
memcpy(dst, dst + pitch, code);
|
|
|
|
}
|
|
|
|
dst += code;
|
|
|
|
} else {
|
|
|
|
code >>= 3;
|
|
|
|
color = codeTable[code];
|
|
|
|
if (color == rleCode) {
|
|
|
|
color = dst[pitch];
|
|
|
|
}
|
|
|
|
*dst++ = code;
|
|
|
|
--unpackedSize;
|
|
|
|
}
|
|
|
|
} while (unpackedSize > 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CUP_Player::handleSRLE(IMAGE *img1, IMAGE *img2, Common::Rect *r, const uint8 *data, uint32 dataSize) {
|
|
|
|
Common::Rect r1;
|
|
|
|
r1.left = READ_LE_UINT16(data);
|
|
|
|
r1.top = READ_LE_UINT16(data + 2);
|
|
|
|
r1.right = READ_LE_UINT16(data + 4);
|
|
|
|
r1.bottom = READ_LE_UINT16(data + 6);
|
|
|
|
const uint8 *codeData = data + 8;
|
|
|
|
uint32 size = READ_LE_UINT32(data + 40);
|
|
|
|
processSRLE(img1->ptr, img2->ptr, codeData, data + 44, size);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CUP_Player::handleWRLE(IMAGE *img, Common::Rect *r, const uint8 *data, uint32 dataSize) {
|
|
|
|
Common::Rect r1;
|
|
|
|
r1.left = READ_LE_UINT16(data); data += 2;
|
|
|
|
r1.top = READ_LE_UINT16(data); data += 2;
|
|
|
|
r1.right = READ_LE_UINT16(data); data += 2;
|
|
|
|
r1.bottom = READ_LE_UINT16(data); data += 2;
|
|
|
|
const uint8 *codeData = data + 8;
|
|
|
|
uint32 size = READ_LE_UINT32(data + 40);
|
|
|
|
processWRLE(img->ptr, codeData, data + 44, size, 0, -img->w);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
} // End of namespace Scumm
|