scummvm/engines/gob/surface.cpp
Sven Hesse 5bec0cbb9d GOB: Make coordinate parameters in Surface::fillRect() signed
And clip to [0, width), [0, height) before drawing.

This fixes bug #6864, which is a regression I introduced in
51fd528fe5 when I changed all
the drawing code to use the Surface class.

I thought that having unsigned coordinates makes sense, but
for some reason, Fascination sets _destSpriteX (which maps
to left in fillRect()) to -1, expecting the drawing code to
clip.
2015-05-07 00:34:02 +02:00

851 lines
20 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.
*
*/
#include "gob/surface.h"
#include "common/system.h"
#include "common/stream.h"
#include "common/util.h"
#include "common/frac.h"
#include "common/textconsole.h"
#include "graphics/primitives.h"
#include "graphics/pixelformat.h"
#include "graphics/surface.h"
#include "image/iff.h"
namespace Gob {
static void plotPixel(int x, int y, int color, void *data) {
Surface *dest = (Surface *)data;
dest->putPixel(x, y, color);
}
Pixel::Pixel(byte *vidMem, uint8 bpp, byte *min, byte *max) :
_vidMem(vidMem), _bpp(bpp), _min(min), _max(max) {
assert((_bpp == 1) || (_bpp == 2) || (_bpp == 4));
assert(_vidMem >= _min);
assert(_vidMem < _max);
}
Pixel &Pixel::operator++() {
_vidMem += _bpp;
return *this;
}
Pixel Pixel::operator++(int x) {
Pixel p = *this;
++(*this);
return p;
}
Pixel &Pixel::operator--() {
_vidMem -= _bpp;
return *this;
}
Pixel Pixel::operator--(int x) {
Pixel p = *this;
--(*this);
return p;
}
Pixel &Pixel::operator+=(int x) {
_vidMem += x * _bpp;
return *this;
}
Pixel &Pixel::operator-=(int x) {
_vidMem -= x * _bpp;
return *this;
}
uint32 Pixel::get() const {
assert(_vidMem >= _min);
assert(_vidMem < _max);
if (_bpp == 1)
return *((byte *) _vidMem);
if (_bpp == 2)
return *((uint16 *) _vidMem);
if (_bpp == 4)
return *((uint32 *) _vidMem);
return 0;
}
void Pixel::set(uint32 p) {
assert(_vidMem >= _min);
assert(_vidMem < _max);
if (_bpp == 1)
*((byte *) _vidMem) = (byte) p;
if (_bpp == 2)
*((uint16 *) _vidMem) = (uint16) p;
if (_bpp == 4)
*((uint32 *) _vidMem) = (uint32) p;
}
bool Pixel::isValid() const {
return (_vidMem >= _min) && (_vidMem < _max);
}
ConstPixel::ConstPixel(const byte *vidMem, uint8 bpp, const byte *min, const byte *max) :
_vidMem(vidMem), _bpp(bpp), _min(min), _max(max) {
assert((_bpp == 1) || (_bpp == 2) || (_bpp == 4));
assert(_vidMem >= _min);
assert(_vidMem < _max);
}
ConstPixel &ConstPixel::operator++() {
_vidMem += _bpp;
return *this;
}
ConstPixel ConstPixel::operator++(int x) {
ConstPixel p = *this;
++(*this);
return p;
}
ConstPixel &ConstPixel::operator--() {
_vidMem -= _bpp;
return *this;
}
ConstPixel ConstPixel::operator--(int x) {
ConstPixel p = *this;
--(*this);
return p;
}
ConstPixel &ConstPixel::operator+=(int x) {
_vidMem += x * _bpp;
return *this;
}
ConstPixel &ConstPixel::operator-=(int x) {
_vidMem -= x * _bpp;
return *this;
}
uint32 ConstPixel::get() const {
assert(_vidMem >= _min);
assert(_vidMem < _max);
if (_bpp == 1)
return *((const byte *) _vidMem);
if (_bpp == 2)
return *((const uint16 *) _vidMem);
if (_bpp == 4)
return *((const uint32 *) _vidMem);
return 0;
}
bool ConstPixel::isValid() const {
return (_vidMem >= _min) && (_vidMem < _max);
}
Surface::Surface(uint16 width, uint16 height, uint8 bpp, byte *vidMem) :
_width(width), _height(height), _bpp(bpp), _vidMem(vidMem) {
assert((_width > 0) && (_height > 0));
assert((_bpp == 1) || (_bpp == 2) || (_bpp == 4));
if (!_vidMem) {
_vidMem = new byte[_bpp * _width * _height];
_ownVidMem = true;
memset(_vidMem, 0, _bpp * _width * _height);
} else
_ownVidMem = false;
}
Surface::Surface(uint16 width, uint16 height, uint8 bpp, const byte *vidMem) :
_width(width), _height(height), _bpp(bpp), _vidMem(0) {
assert((_width > 0) && (_height > 0));
assert((_bpp == 1) || (_bpp == 2) || (_bpp == 4));
_vidMem = new byte[_bpp * _width * _height];
_ownVidMem = true;
memcpy(_vidMem, vidMem, _bpp * _width * _height);
}
Surface::~Surface() {
if (_ownVidMem)
delete[] _vidMem;
}
uint16 Surface::getWidth() const {
return _width;
}
uint16 Surface::getHeight() const {
return _height;
}
uint8 Surface::getBPP() const {
return _bpp;
}
void Surface::resize(uint16 width, uint16 height) {
assert((width > 0) && (height > 0));
if (_ownVidMem)
delete[] _vidMem;
_width = width;
_height = height;
_vidMem = new byte[_bpp * _width * _height];
_ownVidMem = true;
memset(_vidMem, 0, _bpp * _width * _height);
}
void Surface::setBPP(uint8 bpp) {
if (_bpp == bpp)
return;
if (_ownVidMem) {
delete[] _vidMem;
_vidMem = new byte[bpp * _width * _height];
} else
_width = (_width * _bpp) / bpp;
_bpp = bpp;
memset(_vidMem, 0, _bpp * _width * _height);
}
byte *Surface::getData(uint16 x, uint16 y) {
return _vidMem + (y * _width * _bpp) + (x * _bpp);
}
const byte *Surface::getData(uint16 x, uint16 y) const {
return _vidMem + (y * _width * _bpp) + (x * _bpp);
}
Pixel Surface::get(uint16 x, uint16 y) {
byte *vidMem = getData(x, y);
return Pixel(vidMem, _bpp, _vidMem, _vidMem + _height * _width * _bpp);
}
ConstPixel Surface::get(uint16 x, uint16 y) const {
const byte *vidMem = getData(x, y);
return ConstPixel(vidMem, _bpp, _vidMem, _vidMem + _height * _width * _bpp);
}
bool Surface::clipBlitRect(int16 &left, int16 &top, int16 &right, int16 &bottom, int16 &x, int16 &y,
uint16 dWidth, uint16 dHeight, uint16 sWidth, uint16 sHeight) {
if ((x >= dWidth) || (y >= dHeight))
// Nothing to do
return false;
// Just in case those are swapped
if (left > right)
SWAP(left, right);
if (top > bottom)
SWAP(top, bottom);
if ((left >= sWidth) || (top >= sHeight) || (right < 0) || (bottom < 0))
// Nothing to do
return false;
// Adjust from coordinates
if (left < 0) {
x -= left;
left = 0;
}
if (top < 0) {
y -= top;
top = 0;
}
// Adjust to coordinates
if (x < 0) {
left -= x;
x = 0;
}
if (y < 0) {
top -= y;
y = 0;
}
// Limit by source and destination dimensions
right = MIN<int32>(right , MIN<int32>(sWidth , dWidth - x + left) - 1);
bottom = MIN<int32>(bottom, MIN<int32>(sHeight, dHeight - y + top ) - 1);
if ((right < left) || (bottom < top))
// Nothing to do
return false;
// Clip to sane values
right = MAX<int16>(right , 0);
bottom = MAX<int16>(bottom, 0);
return true;
}
void Surface::blit(const Surface &from, int16 left, int16 top, int16 right, int16 bottom,
int16 x, int16 y, int32 transp) {
// Color depths have to fit
assert(_bpp == from._bpp);
// Clip
if (!clipBlitRect(left, top, right, bottom, x, y, _width, _height, from._width, from._height))
return;
// Area to actually copy
uint16 width = right - left + 1;
uint16 height = bottom - top + 1;
if ((width == 0) || (height == 0))
// Nothing to do
return;
if ((left == 0) && (_width == from._width) && (_width == width) && (transp == -1)) {
// If these conditions are met, we can directly use memmove
// Pointers to the blit destination and source start points
byte *dst = getData(x , y);
const byte *src = from.getData(left, top);
memmove(dst, src, width * height * _bpp);
return;
}
if (transp == -1) {
// We don't have to look for transparency => we can use memmove line-wise
// Pointers to the blit destination and source start points
byte *dst = getData(x , y);
const byte *src = from.getData(left, top);
while (height-- > 0) {
memmove(dst, src, width * _bpp);
dst += _width * _bpp;
src += from._width * from._bpp;
}
return;
}
// Otherwise, we have to copy by pixel
// Pointers to the blit destination and source start points
Pixel dst = get(x , y);
ConstPixel src = from.get(left, top);
while (height-- > 0) {
Pixel dstRow = dst;
ConstPixel srcRow = src;
for (uint16 i = 0; i < width; i++, dstRow++, srcRow++)
if (srcRow.get() != ((uint32) transp))
dstRow.set(srcRow.get());
dst += _width;
src += from._width;
}
}
void Surface::blit(const Surface &from, int16 x, int16 y, int32 transp) {
blit(from, 0, 0, from._width - 1, from._height - 1, x, y, transp);
}
void Surface::blit(const Surface &from, int32 transp) {
blit(from, 0, 0, from._width - 1, from._height - 1, 0, 0, transp);
}
void Surface::blitScaled(const Surface &from, int16 left, int16 top, int16 right, int16 bottom,
int16 x, int16 y, Common::Rational scale, int32 transp) {
if (scale == 1) {
// Yeah, "scaled"
blit(from, left, top, right, bottom, x, y, transp);
return;
}
// Color depths have to fit
assert(_bpp == from._bpp);
uint16 dWidth = (uint16) floor((_width / scale).toDouble());
uint16 dHeight = (uint16) floor((_height / scale).toDouble());
int16 clipX = ( int16) floor((x / scale).toDouble());
int16 clipY = ( int16) floor((y / scale).toDouble());
// Clip
if (!clipBlitRect(left, top, right, bottom, clipX, clipY, dWidth, dHeight, from._width, from._height))
return;
// Area to actually copy
uint16 width = right - left + 1;
uint16 height = bottom - top + 1;
if ((width == 0) || (height == 0))
// Nothing to do
return;
width = MIN<int32>((int32) floor((width * scale).toDouble()), _width);
height = MIN<int32>((int32) floor((height * scale).toDouble()), _height);
// Pointers to the blit destination and source start points
byte *dst = getData(x , y);
const byte *src = from.getData(left, top);
frac_t step = scale.getInverse().toFrac();
frac_t posW = 0, posH = 0;
while (height-- > 0) {
byte *dstRow = dst;
const byte *srcRow = src;
posW = 0;
for (uint16 i = 0; i < width; i++, dstRow += _bpp) {
memmove(dstRow, srcRow, _bpp);
posW += step;
while (posW >= ((frac_t) FRAC_ONE)) {
srcRow += from._bpp;
posW -= FRAC_ONE;
}
}
posH += step;
while (posH >= ((frac_t) FRAC_ONE)) {
src += from._width * from._bpp;
posH -= FRAC_ONE;
}
dst += _width * _bpp;
}
}
void Surface::blitScaled(const Surface &from, int16 x, int16 y, Common::Rational scale, int32 transp) {
blitScaled(from, 0, 0, from._width - 1, from._height - 1, x, y, scale, transp);
}
void Surface::blitScaled(const Surface &from, Common::Rational scale, int32 transp) {
blitScaled(from, 0, 0, from._width - 1, from._height - 1, 0, 0, scale, transp);
}
void Surface::fillRect(int16 left, int16 top, int16 right, int16 bottom, uint32 color) {
// Just in case those are swapped
if (left > right)
SWAP(left, right);
if (top > bottom)
SWAP(top, bottom);
if ((left >= _width) || (top >= _height))
// Nothing to do
return;
left = CLIP<int32>(left , 0, _width - 1);
top = CLIP<int32>(top , 0, _height - 1);
right = CLIP<int32>(right , 0, _width - 1);
bottom = CLIP<int32>(bottom, 0, _height - 1);
// Area to actually fill
uint16 width = CLIP<int32>(right - left + 1, 0, _width - left);
uint16 height = CLIP<int32>(bottom - top + 1, 0, _height - top);
if ((width == 0) || (height == 0))
// Nothing to do
return;
if ((left == 0) && (width == _width) && (_bpp == 1)) {
// We can directly use memset
byte *dst = getData(left, top);
memset(dst, (byte) color, width * height);
return;
}
if (_bpp == 1) {
// We can use memset line-wise
byte *dst = getData(left, top);
while (height-- > 0) {
memset(dst, (byte) color, width);
dst += _width;
}
return;
}
assert((_bpp == 2) || (_bpp == 4));
// Otherwise, we have to fill by pixel
Pixel p = get(left, top);
while (height-- > 0) {
for (uint16 i = 0; i < width; i++, ++p)
p.set(color);
p += _width - width;
}
}
void Surface::fill(uint32 color) {
if (_bpp == 1) {
// We can directly use memset
memset(_vidMem, (byte) color, _width * _height);
return;
}
fillRect(0, 0, _width - 1, _height - 1, color);
}
void Surface::clear() {
fill(0);
}
void Surface::shadeRect(uint16 left, uint16 top, uint16 right, uint16 bottom,
uint32 color, uint8 strength) {
if (_bpp == 1) {
// We can't properly shade in paletted mode, fill the rect instead
fillRect(left, top, right, bottom, color);
return;
}
// Just in case those are swapped
if (left > right)
SWAP(left, right);
if (top > bottom)
SWAP(top, bottom);
if ((left >= _width) || (top >= _height))
// Nothing to do
return;
// Area to actually shade
uint16 width = CLIP<int32>(right - left + 1, 0, _width - left);
uint16 height = CLIP<int32>(bottom - top + 1, 0, _height - top);
if ((width == 0) || (height == 0))
// Nothing to do
return;
Graphics::PixelFormat pixelFormat = g_system->getScreenFormat();
uint8 cR, cG, cB;
pixelFormat.colorToRGB(color, cR, cG, cB);
int shadeR = cR * (16 - strength);
int shadeG = cG * (16 - strength);
int shadeB = cB * (16 - strength);
Pixel p = get(left, top);
while (height-- > 0) {
for (uint16 i = 0; i < width; i++, ++p) {
uint8 r, g, b;
pixelFormat.colorToRGB(p.get(), r, g, b);
r = CLIP<int>((shadeR + strength * r) >> 4, 0, 255);
g = CLIP<int>((shadeG + strength * g) >> 4, 0, 255);
b = CLIP<int>((shadeB + strength * b) >> 4, 0, 255);
p.set(pixelFormat.RGBToColor(r, g, b));
}
p += _width - width;
}
}
void Surface::recolor(uint8 from, uint8 to) {
for (Pixel p = get(); p.isValid(); ++p)
if (p.get() == from)
p.set(to);
}
void Surface::putPixel(uint16 x, uint16 y, uint32 color) {
if ((x >= _width) || (y >= _height))
return;
get(x, y).set(color);
}
void Surface::drawLine(uint16 x0, uint16 y0, uint16 x1, uint16 y1, uint32 color) {
Graphics::drawLine(x0, y0, x1, y1, color, &plotPixel, this);
}
void Surface::drawRect(uint16 left, uint16 top, uint16 right, uint16 bottom, uint32 color) {
// Just in case those are swapped
if (left > right)
SWAP(left, right);
if (top > bottom)
SWAP(top, bottom);
if ((left >= _width) || (top >= _height))
// Nothing to do
return;
// Area to actually draw
const uint16 width = CLIP<int32>(right - left + 1, 0, _width - left);
const uint16 height = CLIP<int32>(bottom - top + 1, 0, _height - top);
if ((width == 0) || (height == 0))
// Nothing to do
return;
right = left + width - 1;
bottom = top + height - 1;
drawLine(left , top , left , bottom, color);
drawLine(right, top , right, bottom, color);
drawLine(left , top , right, top , color);
drawLine(left , bottom, right, bottom, color);
}
/*
* The original's version of the Bresenham Algorithm was a bit "unclean"
* and produced strange edges at 45, 135, 225 and 315 degrees, so using the
* version found in the Wikipedia article about the
* "Bresenham's line algorithm" instead
*/
void Surface::drawCircle(uint16 x0, uint16 y0, uint16 radius, uint32 color, int16 pattern) {
int16 f = 1 - radius;
int16 ddFx = 0;
int16 ddFy = -2 * radius;
int16 x = 0;
int16 y = radius;
if (pattern == 0) {
putPixel(x0, y0 + radius, color);
putPixel(x0, y0 - radius, color);
putPixel(x0 + radius, y0, color);
putPixel(x0 - radius, y0, color);
} else
warning("Surface::drawCircle - pattern %d", pattern);
while (x < y) {
if (f >= 0) {
y--;
ddFy += 2;
f += ddFy;
}
x++;
ddFx += 2;
f += ddFx + 1;
switch (pattern) {
case -1:
fillRect(x0 - y, y0 + x, x0 + y, y0 + x, color);
fillRect(x0 - x, y0 + y, x0 + x, y0 + y, color);
fillRect(x0 - y, y0 - x, x0 + y, y0 - x, color);
fillRect(x0 - x, y0 - y, x0 + x, y0 - y, color);
break;
case 0:
putPixel(x0 + x, y0 + y, color);
putPixel(x0 - x, y0 + y, color);
putPixel(x0 + x, y0 - y, color);
putPixel(x0 - x, y0 - y, color);
putPixel(x0 + y, y0 + x, color);
putPixel(x0 - y, y0 + x, color);
putPixel(x0 + y, y0 - x, color);
putPixel(x0 - y, y0 - x, color);
break;
default:
fillRect(x0 + y - pattern, y0 + x - pattern, x0 + y, y0 + x, color);
fillRect(x0 + x - pattern, y0 + y - pattern, x0 + x, y0 + y, color);
fillRect(x0 - y, y0 + x - pattern, x0 - y + pattern, y0 + x, color);
fillRect(x0 - x, y0 + y - pattern, x0 - x + pattern, y0 + y, color);
fillRect(x0 + y - pattern, y0 - x, x0 + y, y0 - x + pattern, color);
fillRect(x0 + x - pattern, y0 - y, x0 + x, y0 - y + pattern, color);
fillRect(x0 - y, y0 - x, x0 - y + pattern, y0 - x + pattern, color);
fillRect(x0 - x, y0 - y, x0 - x + pattern, y0 - y + pattern, color);
break;
}
}
}
void Surface::blitToScreen(uint16 left, uint16 top, uint16 right, uint16 bottom, uint16 x, uint16 y) const {
// Color depths have to fit
assert(g_system->getScreenFormat().bytesPerPixel == _bpp);
uint16 sWidth = g_system->getWidth();
uint16 sHeight = g_system->getHeight();
if ((x >= sWidth) || (y >= sHeight))
// Nothing to do
return;
// Just in case those are swapped
if (left > right)
SWAP(left, right);
if (top > bottom)
SWAP(top, bottom);
if ((left >= _width) || (top >= _height))
// Nothing to do
return;
// Area to actually copy
uint16 width = MAX<int32>(MIN<int32>(MIN<int32>(right - left + 1, _width - left), sWidth - x), 0);
uint16 height = MAX<int32>(MIN<int32>(MIN<int32>(bottom - top + 1, _height - top ), sHeight - y), 0);
if ((width == 0) || (height == 0))
// Nothing to do
return;
// Pointers to the blit destination and source start points
const byte *src = getData(left, top);
g_system->copyRectToScreen(src, _width * _bpp, x, y, width, height);
}
bool Surface::loadImage(Common::SeekableReadStream &stream) {
ImageType type = identifyImage(stream);
if (type == kImageTypeNone)
return false;
return loadImage(stream, type);
}
bool Surface::loadImage(Common::SeekableReadStream &stream, ImageType type) {
if (type == kImageTypeNone)
return false;
switch (type) {
case kImageTypeTGA:
return loadTGA(stream);
case kImageTypeIFF:
return loadIFF(stream);
case kImageTypeBRC:
return loadBRC(stream);
case kImageTypeBMP:
return loadBMP(stream);
case kImageTypeJPEG:
return loadJPEG(stream);
default:
warning("Surface::loadImage(): Unknown image type: %d", (int) type);
return false;
}
return false;
}
ImageType Surface::identifyImage(Common::SeekableReadStream &stream) {
uint32 startPos = stream.pos();
if ((stream.size() - startPos) < 17)
return kImageTypeNone;
char buffer[10];
if (!stream.read(buffer, 10))
return kImageTypeNone;
stream.seek(startPos);
if (!strncmp(buffer , "FORM", 4))
return kImageTypeIFF;
if (!strncmp(buffer + 6, "JFIF", 4))
return kImageTypeJPEG;
if (!strncmp(buffer , "BRC" , 3))
return kImageTypeBRC;
if (!strncmp(buffer , "BM" , 2))
return kImageTypeBMP;
// Try to determine if it's maybe a TGA
stream.skip(12);
uint16 width = stream.readUint16LE();
uint16 height = stream.readUint16LE();
uint8 bpp = stream.readByte();
// Check width, height and bpp for sane values
if ((width == 0) || (height == 0) || (bpp == 0))
return kImageTypeNone;
if ((width > 800) || (height > 600))
return kImageTypeNone;
if ((bpp != 8) && (bpp != 16) && (bpp != 24) && (bpp != 32))
return kImageTypeNone;
// This might be a TGA
return kImageTypeTGA;
}
bool Surface::loadTGA(Common::SeekableReadStream &stream) {
warning("TODO: Surface::loadTGA()");
return false;
}
bool Surface::loadIFF(Common::SeekableReadStream &stream) {
Image::IFFDecoder decoder;
decoder.loadStream(stream);
if (!decoder.getSurface())
return false;
resize(decoder.getSurface()->w, decoder.getSurface()->h);
memcpy(_vidMem, decoder.getSurface()->getPixels(), decoder.getSurface()->w * decoder.getSurface()->h);
return true;
}
bool Surface::loadBRC(Common::SeekableReadStream &stream) {
warning("TODO: Surface::loadBRC()");
return false;
}
bool Surface::loadBMP(Common::SeekableReadStream &stream) {
warning("TODO: Surface::loadBMP()");
return false;
}
bool Surface::loadJPEG(Common::SeekableReadStream &stream) {
warning("TODO: Surface::loadJPEG()");
return false;
}
} // End of namespace Gob