more cleanup; fixes out of bound access to _charsetData (but I am not sure if that will be visible anywhere

svn-id: r5833
This commit is contained in:
Max Horn 2002-12-04 15:00:35 +00:00
parent bb2d559d13
commit 77a64e0217
4 changed files with 309 additions and 327 deletions

@ -114,17 +114,18 @@ class Gdi {
public:
Scumm *_vm;
protected:
byte *_readPtr;
uint _readOffs;
public:
int _numZBuffer;
int _imgBufOffs[5];
byte _disable_zbuffer;
int32 _numStrips;
public:
int16 _mask_top, _mask_bottom, _mask_right, _mask_left;
protected:
byte *_readPtr;
uint _readOffs;
bool _useOrDecompress;
int _numLinesToProcess;
int _tempNumLines;
@ -135,9 +136,6 @@ protected:
int16 _drawMouseX;
int16 _drawMouseY;
public:
int16 _mask_top, _mask_bottom, _mask_right, _mask_left;
protected:
byte _mouseColors[4];
byte _mouseColor;
byte _mouseClipMask1, _mouseClipMask2, _mouseClipMask3;
@ -169,14 +167,6 @@ protected:
void unkDecode10();
void unkDecode11();
public:
void drawBitmap(byte *ptr, VirtScreen *vs, int x, int y, int h, int stripnr, int numstrip, byte flag);
void clearUpperMask();
void disableZBuffer() { _disable_zbuffer++; }
void enableZBuffer() { _disable_zbuffer--; }
protected:
void draw8ColWithMasking(byte *dst, byte *src, int height, byte *mask);
void draw8Col(byte *dst, byte *src, int height);
void clear8ColWithMasking(byte *dst, int height, byte *mask);
@ -185,15 +175,21 @@ protected:
void decompressMaskImg(byte *dst, byte *src, int height);
public:
void drawBitmap(byte *ptr, VirtScreen *vs, int x, int y, int h, int stripnr, int numstrip, byte flag);
void clearUpperMask();
void disableZBuffer() { _disable_zbuffer++; }
void enableZBuffer() { _disable_zbuffer--; }
void resetBackground(int top, int bottom, int strip);
void drawStripToScreen(VirtScreen *vs, int x, int w, int t, int b);
void updateDirtyScreen(VirtScreen *vs);
enum DrawBitmapFlags {
dbAllowMaskOr = 1,
dbDrawMaskOnAll = 2,
dbClear = 4
};
void resetBackground(int top, int bottom, int strip);
void drawStripToScreen(VirtScreen *vs, int x, int w, int t, int b);
void updateDirtyScreen(VirtScreen *vs);
};
#endif

@ -455,6 +455,8 @@ void Scumm::loadCharset(int no)
// ensureResourceLoaded(rtCharset, no);
ptr = getResourceAddress(rtCharset, no);
if (_features & GF_SMALL_HEADER)
ptr -= 12;
for (i = 0; i < 15; i++) {
_charsetData[no][i + 1] = ptr[i + 14];

@ -171,7 +171,12 @@ enum {
#define _roomFileOffsets res.roomoffs[rtRoom]
struct CharsetRenderer {
class CharsetRenderer {
protected:
byte _curId;
byte *_fontPtr;
public:
Scumm *_vm;
int _top;
int _drawTop;
@ -183,7 +188,6 @@ struct CharsetRenderer {
bool _blitAlso;
int _strLeft, _strRight, _strTop, _strBottom;
byte _curId;
int _xpos2, _ypos2;
@ -209,6 +213,12 @@ struct CharsetRenderer {
int getSpacing(byte chr, byte *charset);
int getStringWidth(int a, byte *str, int pos);
void addLinebreaks(int a, byte *str, int pos, int maxwidth);
void setCurID(byte id);
int getCurID() { return _curId; }
byte *getFontPtr() { return _fontPtr; }
byte *getFontPtr(byte id);
};
#define ARRAY_HDR_SIZE 6

@ -27,6 +27,51 @@
#include "verbs.h"
#include "scumm/sound.h"
void CharsetRenderer::setCurID(byte id) {
_curId = id;
_fontPtr = getFontPtr(id);
}
byte *CharsetRenderer::getFontPtr(byte id)
{
byte *ptr = _vm->getResourceAddress(rtCharset, id);
assert(ptr);
if (_vm->_features & GF_SMALL_HEADER)
ptr += 17;
else
ptr += 29;
return ptr;
}
// do spacing for variable width old-style font
int CharsetRenderer::getSpacing(byte chr, byte *charset)
{
int spacing = 0;
if (_vm->_features & GF_OLD256) {
spacing = *(charset - 11 + chr);
} else {
int offs = READ_LE_UINT32(charset + chr * 4 + 4);
if (offs) {
spacing = charset[offs];
if (charset[offs + 2] >= 0x80) {
spacing += charset[offs + 2] - 0x100;
} else {
spacing += charset[offs + 2];
}
}
}
// FIXME - this fixes the inventory icons in Zak256/Indy3
// see bug #613109.
// chars 1,2: up arrow chars 3,4: down arrow
if ((_vm->_gameId == GID_ZAK256 || _vm->_gameId == GID_INDY3_256)
&& (chr >= 1 && chr <= 4))
spacing = 6;
return spacing;
}
int CharsetRenderer::getStringWidth(int arg, byte *text, int pos)
{
byte *ptr;
@ -34,9 +79,7 @@ int CharsetRenderer::getStringWidth(int arg, byte *text, int pos)
byte chr;
width = 1;
ptr = _vm->getResourceAddress(rtCharset, _curId) + 29;
if (_vm->_features & GF_SMALL_HEADER)
ptr -= 12;
ptr = _fontPtr;
while ((chr = text[pos++]) != 0) {
if (chr == 0xD)
@ -63,9 +106,7 @@ int CharsetRenderer::getStringWidth(int arg, byte *text, int pos)
if (chr == 14) {
int set = text[pos] | (text[pos + 1] << 8);
pos += 2;
ptr = _vm->getResourceAddress(rtCharset, set) + 29;
if (_vm->_features & GF_SMALL_HEADER)
ptr -= 12;
ptr = getFontPtr(set);
continue;
}
}
@ -82,9 +123,7 @@ void CharsetRenderer::addLinebreaks(int a, byte *str, int pos, int maxwidth)
byte *ptr;
byte chr;
ptr = _vm->getResourceAddress(rtCharset, _curId) + 29;
if (_vm->_features & GF_SMALL_HEADER)
ptr -= 12;
ptr = _fontPtr;
while ((chr = str[pos++]) != 0) {
if (chr == '@')
@ -117,9 +156,7 @@ void CharsetRenderer::addLinebreaks(int a, byte *str, int pos, int maxwidth)
if (chr == 14) {
int set = str[pos] | (str[pos + 1] << 8);
pos += 2;
ptr = _vm->getResourceAddress(rtCharset, set) + 29;
if (_vm->_features & GF_SMALL_HEADER)
ptr -= 12;
ptr = getFontPtr(set);
continue;
}
}
@ -139,6 +176,220 @@ void CharsetRenderer::addLinebreaks(int a, byte *str, int pos, int maxwidth)
}
}
void CharsetRenderer::printCharOld(int chr)
{ // Indy3 / Zak256
VirtScreen *vs;
byte *char_ptr, *dest_ptr;
unsigned int buffer = 0, mask = 0, x = 0, y = 0;
unsigned char color;
_vm->checkRange(_vm->_maxCharsets - 1, 0, _curId, "Printing with bad charset %d");
if ((vs = _vm->findVirtScreen(_top)) == NULL)
return;
if (chr == '@')
return;
if (_firstChar) {
_strLeft = _left;
_strTop = _top;
_strRight = _left;
_strBottom = _top;
_firstChar = false;
}
char_ptr = _fontPtr + 207 + (chr + 1) * 8;
dest_ptr = vs->screenPtr + vs->xstart + (_top - vs->topline) * _vm->_realWidth + _left;
_vm->updateDirtyRect(vs->number, _left, _left + 8, _top - vs->topline, _top - vs->topline + 8, 0);
for (y = 0; y < 8; y++) {
for (x = 0; x < 8; x++) {
if ((mask >>= 1) == 0) {
buffer = *char_ptr++;
mask = 0x80;
}
color = ((buffer & mask) != 0);
if (color)
*(dest_ptr + y * _vm->_realWidth + x) = _color;
}
}
// FIXME
_left += getSpacing(chr, _fontPtr);
if (_left > _strRight)
_strRight = _left;
if (_top + 8 > _strBottom)
_strBottom = _top + 8;
}
void CharsetRenderer::printChar(int chr)
{
int d, right;
VirtScreen *vs;
_vm->checkRange(_vm->_maxCharsets - 1, 1, _curId, "Printing with bad charset %d");
if ((vs = _vm->findVirtScreen(_top)) == NULL)
return;
if (chr == '@')
return;
_bpp = *_fontPtr;
_colorMap[1] = _color;
_charOffs = READ_LE_UINT32(_fontPtr + chr * 4 + 4);
if (!_charOffs)
return;
assert(_charOffs < 0x10000);
_charPtr = _fontPtr + _charOffs;
_width = _charPtr[0];
_height = _charPtr[1];
if (_firstChar) {
_strLeft = 0;
_strTop = 0;
_strRight = 0;
_strBottom = 0;
}
if (_disableOffsX) {
_offsX = 0;
} else {
d = _charPtr[2];
if (d >= 0x80)
d -= 0x100;
_offsX = d;
}
d = _charPtr[3];
if (d >= 0x80)
d -= 0x100;
_offsY = d;
_top += _offsY;
_left += _offsX;
right = _left + _width;
if (right > _right + 1 || _left < 0) {
_left = right;
_top -= _offsY;
return;
}
_disableOffsX = false;
if (_firstChar) {
_strLeft = _left;
_strTop = _top;
_strRight = _left;
_strBottom = _top;
_firstChar = false;
}
if (_left < _strLeft)
_strLeft = _left;
if (_top < _strTop)
_strTop = _top;
_drawTop = _top - vs->topline;
if (_drawTop < 0)
_drawTop = 0;
_bottom = _drawTop + _height + _offsY;
_vm->updateDirtyRect(vs->number, _left, right, _drawTop, _bottom, 0);
if (vs->number != 0)
_blitAlso = false;
if (vs->number == 0 && _blitAlso == 0)
_hasMask = true;
_virtScreenHeight = vs->height;
_charPtr += 4;
byte *mask = _vm->getResourceAddress(rtBuffer, 9)
+ _drawTop * _vm->gdi._numStrips + _left / 8 + _vm->_screenStartStrip;
byte *dst = vs->screenPtr + vs->xstart + _drawTop * _vm->_realWidth + _left;
if (_blitAlso) {
byte *back = dst;
dst = _vm->getResourceAddress(rtBuffer, vs->number + 5)
+ vs->xstart + _drawTop * _vm->_realWidth + _left;
drawBits(dst, mask);
_vm->blit(back, dst, _width, _height);
} else {
drawBits(dst, mask);
}
_left += _width;
if (_left > _strRight)
_strRight = _left;
if (_top + _height > _strBottom)
_strBottom = _top + _height;
_top -= _offsY;
}
void CharsetRenderer::drawBits(byte *dst, byte *mask)
{
bool usemask;
byte maskmask;
int y, x;
int maskpos;
int color;
byte numbits, bits;
usemask = (_vm->_curVirtScreen->number == 0 && _ignoreCharsetMask == 0);
bits = *_charPtr++;
numbits = 8;
y = 0;
for (y = 0; y < _height && y + _drawTop < _virtScreenHeight;) {
maskmask = revBitMask[_left & 7];
maskpos = 0;
for (x = 0; x < _width; x++) {
color = (bits >> (8 - _bpp)) & 0xFF;
if (color) {
if (usemask) {
mask[maskpos] |= maskmask;
}
*dst = _colorMap[color];
}
dst++;
bits <<= _bpp;
if ((numbits -= _bpp) == 0) {
bits = *_charPtr++;
numbits = 8;
}
if ((maskmask >>= 1) == 0) {
maskmask = 0x80;
maskpos++;
}
}
dst += _vm->_realWidth - _width;
mask += _vm->gdi._numStrips;
y++;
}
}
void Scumm::unkMessage1()
{
byte buffer[100];
@ -243,10 +494,12 @@ void Scumm::CHARSET_1()
charset._top = _string[0].ypos;
charset._left = _string[0].xpos;
charset._startLeft = _string[0].xpos;
charset._curId = _string[0].charset;
if (a && a->charset)
charset._curId = a->charset;
charset.setCurID(a->charset);
else
charset.setCurID(_string[0].charset);
charset._center = _string[0].center;
charset._right = _string[0].right;
@ -254,10 +507,7 @@ void Scumm::CHARSET_1()
if (!(_features & GF_OLD256)) // FIXME
for (i = 0; i < 4; i++)
if (_features & GF_SMALL_HEADER)
charset._colorMap[i] = _charsetData[charset._curId][i - 12]; // FIXME - do we really want to access index -12 to -9 ?
else
charset._colorMap[i] = _charsetData[charset._curId][i];
charset._colorMap[i] = _charsetData[charset.getCurID()][i];
if (_keepText) {
charset._strLeft = gdi._mask_left;
@ -342,10 +592,7 @@ void Scumm::CHARSET_1()
if (charset._center) {
charset._xpos2 -= charset.getStringWidth(0, buffer, 0) >> 1;
}
if (_features & GF_SMALL_HEADER)
charset._ypos2 += getResourceAddress(rtCharset, charset._curId)[30 - 12];
else
charset._ypos2 += getResourceAddress(rtCharset, charset._curId)[30];
charset._ypos2 += charset.getFontPtr()[1];
charset._disableOffsX = true;
continue;
}
@ -416,23 +663,13 @@ void Scumm::CHARSET_1()
buffer += 2;
break;
case 14: {
int oldy;
if (_features & GF_SMALL_HEADER)
oldy = getResourceAddress(rtCharset, charset._curId)[30 - 12];
else
oldy = getResourceAddress(rtCharset, charset._curId)[30];
int oldy = charset.getFontPtr()[1];
charset._curId = *buffer++;
charset.setCurID(*buffer++);
buffer += 2;
for (i = 0; i < 4; i++)
if (_features & GF_SMALL_HEADER)
charset._colorMap[i] = _charsetData[charset._curId][i - 12]; // FIXME - do we really want to access index -12 to -9 ?
else
charset._colorMap[i] = _charsetData[charset._curId][i];
if (_features & GF_SMALL_HEADER)
charset._ypos2 -= getResourceAddress(rtCharset, charset._curId)[30 - 12] - oldy;
else
charset._ypos2 -= getResourceAddress(rtCharset, charset._curId)[30] - oldy;
charset._colorMap[i] = _charsetData[charset.getCurID()][i];
charset._ypos2 -= charset.getFontPtr()[1] - oldy;
break;
}
default:
@ -480,7 +717,7 @@ void Scumm::description()
charset._xpos2 = _string[0].xpos;
charset._ypos2 = _string[0].ypos;
charset._disableOffsX = charset._firstChar = true;
charset._curId = 3;
charset.setCurID(3);
charset._center = false;
charset._color = 15;
// FIXME: _talkdelay = 1 - display description, not correct ego actor talking,
@ -527,7 +764,7 @@ void Scumm::drawDescString(byte *msg)
charset._right = _realWidth - 1;
charset._xpos2 = _string[0].xpos;
charset._ypos2 = _string[0].ypos;
charset._curId = _string[0].charset;
charset.setCurID(_string[0].charset);
charset._center = _string[0].center;
charset._color = _string[0].color;
@ -566,7 +803,7 @@ void Scumm::drawDescString(byte *msg)
void Scumm::drawString(int a)
{
byte buf[256];
byte *charsetptr, *space;
byte *space;
int i;
byte fontHeight = 0, chr;
uint color;
@ -576,26 +813,17 @@ void Scumm::drawString(int a)
charset._startLeft = charset._left = _string[a].xpos;
charset._top = _string[a].ypos;
charset._curId = _string[a].charset;
charset.setCurID(_string[a].charset);
charset._center = _string[a].center;
charset._right = _string[a].right;
charset._color = _string[a].color;
charset._disableOffsX = charset._firstChar = true;
if (!(_features & GF_OLD256)) {
charsetptr = getResourceAddress(rtCharset, charset._curId);
assert(charsetptr);
charsetptr += 29;
if (_features & GF_SMALL_HEADER)
charsetptr -= 12;
for (i = 0; i < 4; i++)
if (_features & GF_SMALL_HEADER)
charset._colorMap[i] = _charsetData[charset._curId][i - 12]; // FIXME - do we really want to access index -12 to -9 ?
else
charset._colorMap[i] = _charsetData[charset._curId][i];
charset._colorMap[i] = _charsetData[charset.getCurID()][i];
fontHeight = charsetptr[1];
fontHeight = charset.getFontPtr()[1];
}
_msgPtrToAdd = buf;
@ -873,262 +1101,8 @@ void Scumm::initCharset(int charsetno)
_string[1].t_charset = charsetno;
for (i = 0; i < 16; i++)
if (_features & GF_SMALL_HEADER)
charset._colorMap[i] = _charsetData[charset._curId][i - 12]; // FIXME - do we really want to access index -12 to -9 ?
else
charset._colorMap[i] = _charsetData[charset._curId][i];
charset._colorMap[i] = _charsetData[charset.getCurID()][i];
}
void CharsetRenderer::printCharOld(int chr)
{ // Indy3 / Zak256
VirtScreen *vs;
byte *char_ptr, *dest_ptr;
unsigned int buffer = 0, mask = 0, x = 0, y = 0;
unsigned char color;
_vm->checkRange(_vm->_maxCharsets - 1, 0, _curId, "Printing with bad charset %d");
if ((vs = _vm->findVirtScreen(_top)) == NULL)
return;
if (chr == '@')
return;
byte *ptr = _vm->getResourceAddress(rtCharset, _curId);
if (_firstChar) {
_strLeft = _left;
_strTop = _top;
_strRight = _left;
_strBottom = _top;
_firstChar = false;
}
char_ptr = ptr + 224 + (chr + 1) * 8;
dest_ptr = vs->screenPtr + vs->xstart + (_top - vs->topline) * _vm->_realWidth + _left;
_vm->updateDirtyRect(vs->number, _left, _left + 8, _top - vs->topline, _top - vs->topline + 8, 0);
for (y = 0; y < 8; y++) {
for (x = 0; x < 8; x++) {
if ((mask >>= 1) == 0) {
buffer = *char_ptr++;
mask = 0x80;
}
color = ((buffer & mask) != 0);
if (color)
*(dest_ptr + y * _vm->_realWidth + x) = _color;
}
}
// FIXME
_left += getSpacing(chr, ptr + 29 - 12);
if (_left > _strRight)
_strRight = _left;
if (_top + 8 > _strBottom)
_strBottom = _top + 8;
}
void CharsetRenderer::printChar(int chr)
{
int d, right;
VirtScreen *vs;
_vm->checkRange(_vm->_maxCharsets - 1, 1, _curId, "Printing with bad charset %d");
if ((vs = _vm->findVirtScreen(_top)) == NULL)
return;
if (chr == '@')
return;
byte *ptr = _vm->getResourceAddress(rtCharset, _curId) + 29;
if (_vm->_features & GF_SMALL_HEADER)
ptr -= 12;
_bpp = *ptr;
_colorMap[1] = _color;
_charOffs = READ_LE_UINT32(ptr + chr * 4 + 4);
if (!_charOffs)
return;
assert(_charOffs < 0x10000);
_charPtr = ptr + _charOffs;
_width = _charPtr[0];
_height = _charPtr[1];
if (_firstChar) {
_strLeft = 0;
_strTop = 0;
_strRight = 0;
_strBottom = 0;
}
if (_disableOffsX) {
_offsX = 0;
} else {
d = _charPtr[2];
if (d >= 0x80)
d -= 0x100;
_offsX = d;
}
d = _charPtr[3];
if (d >= 0x80)
d -= 0x100;
_offsY = d;
_top += _offsY;
_left += _offsX;
right = _left + _width;
if (right > _right + 1 || _left < 0) {
_left = right;
_top -= _offsY;
return;
}
_disableOffsX = false;
if (_firstChar) {
_strLeft = _left;
_strTop = _top;
_strRight = _left;
_strBottom = _top;
_firstChar = false;
}
if (_left < _strLeft)
_strLeft = _left;
if (_top < _strTop)
_strTop = _top;
_drawTop = _top - vs->topline;
if (_drawTop < 0)
_drawTop = 0;
_bottom = _drawTop + _height + _offsY;
_vm->updateDirtyRect(vs->number, _left, right, _drawTop, _bottom, 0);
if (vs->number != 0)
_blitAlso = false;
if (vs->number == 0 && _blitAlso == 0)
_hasMask = true;
byte *_backbuff_ptr;
byte *_mask_ptr;
byte *_dest_ptr;
_dest_ptr = _backbuff_ptr = vs->screenPtr + vs->xstart + _drawTop * _vm->_realWidth + _left;
if (_blitAlso) {
_dest_ptr = _vm->getResourceAddress(rtBuffer, vs->number + 5)
+ vs->xstart + _drawTop * _vm->_realWidth + _left;
}
_mask_ptr = _vm->getResourceAddress(rtBuffer, 9)
+ _drawTop * _vm->gdi._numStrips + _left / 8 + _vm->_screenStartStrip;
_virtScreenHeight = vs->height;
_charPtr += 4;
drawBits(_dest_ptr, _mask_ptr);
if (_blitAlso)
_vm->blit(_backbuff_ptr, _dest_ptr, _width, _height);
_left += _width;
if (_left > _strRight)
_strRight = _left;
if (_top + _height > _strBottom)
_strBottom = _top + _height;
_top -= _offsY;
}
void CharsetRenderer::drawBits(byte *dst, byte *mask)
{
bool usemask;
byte maskmask;
int y, x;
int maskpos;
int color;
byte numbits, bits;
usemask = (_vm->_curVirtScreen->number == 0 && _ignoreCharsetMask == 0);
bits = *_charPtr++;
numbits = 8;
y = 0;
for (y = 0; y < _height && y + _drawTop < _virtScreenHeight;) {
maskmask = revBitMask[_left & 7];
maskpos = 0;
for (x = 0; x < _width; x++) {
color = (bits >> (8 - _bpp)) & 0xFF;
if (color) {
if (usemask) {
mask[maskpos] |= maskmask;
}
*dst = _colorMap[color];
}
dst++;
bits <<= _bpp;
if ((numbits -= _bpp) == 0) {
bits = *_charPtr++;
numbits = 8;
}
if ((maskmask >>= 1) == 0) {
maskmask = 0x80;
maskpos++;
}
}
dst += _vm->_realWidth - _width;
mask += _vm->gdi._numStrips;
y++;
}
}
// do spacing for variable width old-style font
int CharsetRenderer::getSpacing(byte chr, byte *charset)
{
int spacing = 0;
if (_vm->_features & GF_OLD256) {
spacing = *(charset - 11 + chr);
} else {
int offs = READ_LE_UINT32(charset + chr * 4 + 4);
if (offs) {
spacing = charset[offs];
if (charset[offs + 2] >= 0x80) {
spacing += charset[offs + 2] - 0x100;
} else {
spacing += charset[offs + 2];
}
}
}
// FIXME - this fixes the inventory icons in Zak256/Indy3
// see bug #613109.
// chars 1,2: up arrow chars 3,4: down arrow
if ((_vm->_gameId == GID_ZAK256 || _vm->_gameId == GID_INDY3_256)
&& (chr >= 1 && chr <= 4))
spacing = 6;
return spacing;
}
void Scumm::loadLanguageBundle() {
File file;