2007-05-30 21:56:52 +00:00
|
|
|
/* ScummVM - Graphic Adventure Engine
|
2006-02-22 22:40:53 +00:00
|
|
|
*
|
2007-05-30 21:56:52 +00:00
|
|
|
* 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.
|
2006-02-22 22:40:53 +00:00
|
|
|
*
|
|
|
|
* 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 "cine/cine.h"
|
2006-02-25 01:18:01 +00:00
|
|
|
#include "cine/various.h"
|
2009-03-11 21:03:09 +00:00
|
|
|
#include "cine/pal.h"
|
2006-02-22 22:40:53 +00:00
|
|
|
|
2006-02-25 00:26:14 +00:00
|
|
|
namespace Cine {
|
|
|
|
|
2008-08-11 21:45:47 +00:00
|
|
|
Common::Array<PalEntry> palArray;
|
2007-12-14 19:21:19 +00:00
|
|
|
static byte paletteBuffer1[16];
|
|
|
|
static byte paletteBuffer2[16];
|
2006-02-22 22:40:53 +00:00
|
|
|
|
|
|
|
void loadPal(const char *fileName) {
|
|
|
|
char buffer[20];
|
|
|
|
|
|
|
|
removeExtention(buffer, fileName);
|
|
|
|
|
|
|
|
strcat(buffer, ".PAL");
|
2008-08-11 21:45:47 +00:00
|
|
|
palArray.clear();
|
2006-02-22 22:40:53 +00:00
|
|
|
|
2007-05-30 18:43:28 +00:00
|
|
|
Common::File palFileHandle;
|
|
|
|
if (!palFileHandle.open(buffer))
|
|
|
|
error("loadPal(): Cannot open file %s", fileName);
|
2006-02-22 22:40:53 +00:00
|
|
|
|
2009-01-01 15:06:43 +00:00
|
|
|
uint16 palEntriesCount = palFileHandle.readUint16LE();
|
2006-02-27 21:25:59 +00:00
|
|
|
palFileHandle.readUint16LE(); // entry size
|
2007-12-14 19:21:19 +00:00
|
|
|
|
2008-08-11 21:45:47 +00:00
|
|
|
palArray.resize(palEntriesCount);
|
|
|
|
for (uint i = 0; i < palArray.size(); ++i) {
|
|
|
|
palFileHandle.read(palArray[i].name, 10);
|
|
|
|
palFileHandle.read(palArray[i].pal1, 16);
|
|
|
|
palFileHandle.read(palArray[i].pal2, 16);
|
2006-02-27 21:25:59 +00:00
|
|
|
}
|
2007-05-30 18:43:28 +00:00
|
|
|
palFileHandle.close();
|
2006-02-22 22:40:53 +00:00
|
|
|
}
|
|
|
|
|
2006-02-23 09:12:21 +00:00
|
|
|
int16 findPaletteFromName(const char *fileName) {
|
2006-02-22 22:40:53 +00:00
|
|
|
char buffer[10];
|
2006-02-23 09:12:21 +00:00
|
|
|
uint16 position = 0;
|
|
|
|
uint16 i;
|
2006-02-22 22:40:53 +00:00
|
|
|
|
|
|
|
strcpy(buffer, fileName);
|
|
|
|
|
|
|
|
while (position < strlen(fileName)) {
|
|
|
|
if (buffer[position] > 'a' && buffer[position] < 'z') {
|
2006-02-27 21:25:59 +00:00
|
|
|
buffer[position] += 'A' - 'a';
|
2006-02-22 22:40:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
position++;
|
|
|
|
}
|
|
|
|
|
2008-08-11 21:45:47 +00:00
|
|
|
for (i = 0; i < palArray.size(); i++) {
|
|
|
|
if (!strcmp(buffer, palArray[i].name)) {
|
2006-02-22 22:40:53 +00:00
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void loadRelatedPalette(const char *fileName) {
|
|
|
|
char localName[16];
|
2006-03-23 03:45:52 +00:00
|
|
|
byte i;
|
2006-02-23 09:12:21 +00:00
|
|
|
int16 paletteIndex;
|
2006-02-22 22:40:53 +00:00
|
|
|
|
|
|
|
removeExtention(localName, fileName);
|
|
|
|
|
|
|
|
paletteIndex = findPaletteFromName(localName);
|
|
|
|
|
|
|
|
if (paletteIndex == -1) {
|
2006-02-23 18:47:28 +00:00
|
|
|
for (i = 0; i < 16; i++) { // generate default palette
|
2006-02-22 22:40:53 +00:00
|
|
|
paletteBuffer1[i] = paletteBuffer2[i] = (i << 4) + i;
|
|
|
|
}
|
|
|
|
} else {
|
2008-08-11 21:45:47 +00:00
|
|
|
assert(paletteIndex < (int32)palArray.size());
|
|
|
|
memcpy(paletteBuffer1, palArray[paletteIndex].pal1, 16);
|
|
|
|
memcpy(paletteBuffer2, palArray[paletteIndex].pal2, 16);
|
2006-02-22 22:40:53 +00:00
|
|
|
}
|
|
|
|
}
|
2006-02-25 00:26:14 +00:00
|
|
|
|
2008-05-24 22:11:41 +00:00
|
|
|
void palRotate(uint16 *pal, byte a, byte b, byte c) {
|
|
|
|
assert(pal);
|
|
|
|
|
|
|
|
if (c == 1) {
|
|
|
|
uint16 currentColor = pal[b];
|
|
|
|
|
|
|
|
for (int i = b; i > a; i--) {
|
|
|
|
pal[i] = pal[i - 1];
|
|
|
|
}
|
|
|
|
|
|
|
|
pal[a] = currentColor;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void palRotate(byte *pal, byte a, byte b, byte c) {
|
|
|
|
assert(pal);
|
|
|
|
|
|
|
|
if (c == 1) {
|
|
|
|
byte currentR = pal[3 * b + 0];
|
|
|
|
byte currentG = pal[3 * b + 1];
|
|
|
|
byte currentB = pal[3 * b + 2];
|
|
|
|
|
|
|
|
for (int i = b; i > a; i--) {
|
|
|
|
pal[3 * i + 0] = pal[3 * (i - 1) + 0];
|
|
|
|
pal[3 * i + 1] = pal[3 * (i - 1) + 1];
|
|
|
|
pal[3 * i + 2] = pal[3 * (i - 1) + 2];
|
|
|
|
}
|
|
|
|
|
|
|
|
pal[3 * a + 0] = currentR;
|
|
|
|
pal[3 * a + 1] = currentG;
|
|
|
|
pal[3 * a + 2] = currentB;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16 transformColor(uint16 baseColor, int r, int g, int b) {
|
|
|
|
int8 oriR = CLIP( (baseColor & 0x007) + b, 0, 7);
|
|
|
|
int8 oriG = CLIP(((baseColor & 0x070) >> 4) + g, 0, 7);
|
|
|
|
int8 oriB = CLIP(((baseColor & 0x700) >> 8) + r, 0, 7);
|
|
|
|
|
|
|
|
return oriR | (oriG << 4) | (oriB << 8);
|
|
|
|
}
|
|
|
|
|
|
|
|
void transformPaletteRange(uint16 *dstPal, uint16 *srcPal, int startColor, int stopColor, int r, int g, int b) {
|
|
|
|
assert(srcPal && dstPal);
|
|
|
|
|
2009-03-10 22:09:10 +00:00
|
|
|
for (int i = startColor; i <= stopColor; i++)
|
2008-05-24 22:11:41 +00:00
|
|
|
dstPal[i] = transformColor(srcPal[i], r, g, b);
|
|
|
|
}
|
|
|
|
|
|
|
|
void transformPaletteRange(byte *dstPal, byte *srcPal, int startColor, int stopColor, int r, int g, int b) {
|
|
|
|
assert(srcPal && dstPal);
|
|
|
|
|
|
|
|
for (int i = startColor; i <= stopColor; i++) {
|
2008-11-18 19:54:44 +00:00
|
|
|
dstPal[3 * i + 0] = CLIP(srcPal[3 * i + 0] + r * 36, 0, 252);
|
|
|
|
dstPal[3 * i + 1] = CLIP(srcPal[3 * i + 1] + g * 36, 0, 252);
|
|
|
|
dstPal[3 * i + 2] = CLIP(srcPal[3 * i + 2] + b * 36, 0, 252);
|
2008-05-24 22:11:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-03-16 20:24:28 +00:00
|
|
|
/*! \brief Shift byte to the left by given amount (Handles negative shifting amounts too, otherwise this would be trivial). */
|
2009-03-14 22:36:05 +00:00
|
|
|
byte shiftByteLeft(const byte value, const signed shiftLeft) {
|
|
|
|
if (shiftLeft >= 0)
|
|
|
|
return value << shiftLeft;
|
|
|
|
else // right shift with negative shiftLeft values
|
|
|
|
return value >> abs(shiftLeft);
|
|
|
|
}
|
|
|
|
|
2009-03-16 20:24:28 +00:00
|
|
|
/*! \brief Is given endian type big endian? (Handles native endian type too, otherwise this would be trivial). */
|
2009-03-16 21:30:15 +00:00
|
|
|
bool isBigEndian(const EndianType endian) {
|
|
|
|
assert(endian == CINE_NATIVE_ENDIAN || endian == CINE_LITTLE_ENDIAN || endian == CINE_BIG_ENDIAN);
|
2009-03-16 20:24:28 +00:00
|
|
|
|
|
|
|
// Handle explicit little and big endian types here
|
2009-03-16 21:30:15 +00:00
|
|
|
if (endian != CINE_NATIVE_ENDIAN) {
|
|
|
|
return (endian == CINE_BIG_ENDIAN);
|
2009-03-16 20:24:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Handle native endian type here
|
|
|
|
#if defined(SCUMM_BIG_ENDIAN)
|
|
|
|
return true;
|
|
|
|
#elif defined(SCUMM_LITTLE_ENDIAN)
|
|
|
|
return false;
|
|
|
|
#else
|
|
|
|
#error No endianness defined
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/*! \brief Calculate byte position of given bit position in a multibyte variable using defined endianness. */
|
|
|
|
int bytePos(const int bitPos, const int numBytes, const bool bigEndian) {
|
|
|
|
if (bigEndian)
|
|
|
|
return (numBytes - 1) - (bitPos / 8);
|
|
|
|
else // little endian
|
|
|
|
return bitPos / 8;
|
|
|
|
}
|
|
|
|
|
2008-11-20 22:16:18 +00:00
|
|
|
// a.k.a. palRotate
|
2009-03-10 22:09:10 +00:00
|
|
|
Palette &Palette::rotateRight(byte firstIndex, byte lastIndex) {
|
2009-03-10 20:37:54 +00:00
|
|
|
const Color lastColor = _colors[lastIndex];
|
2008-11-20 22:16:18 +00:00
|
|
|
|
2009-03-10 22:09:10 +00:00
|
|
|
for (int i = lastIndex; i > firstIndex; i--)
|
2009-03-10 20:37:54 +00:00
|
|
|
_colors[i] = _colors[i - 1];
|
2008-11-20 22:16:18 +00:00
|
|
|
|
2009-03-10 20:37:54 +00:00
|
|
|
_colors[firstIndex] = lastColor;
|
2008-11-20 22:16:18 +00:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2009-03-10 20:37:54 +00:00
|
|
|
uint Palette::colorCount() const {
|
|
|
|
return _colors.size();
|
2008-11-20 22:16:18 +00:00
|
|
|
}
|
|
|
|
|
2009-03-24 21:45:35 +00:00
|
|
|
Palette &Palette::fillWithBlack() {
|
|
|
|
for (uint i = 0; i < _colors.size(); i++) {
|
|
|
|
_colors[i].r = 0;
|
|
|
|
_colors[i].g = 0;
|
|
|
|
_colors[i].b = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2009-03-16 21:30:15 +00:00
|
|
|
EndianType Palette::endianType() const {
|
2009-03-16 20:57:17 +00:00
|
|
|
return (_bigEndian ? CINE_BIG_ENDIAN : CINE_LITTLE_ENDIAN);
|
2009-03-16 20:24:28 +00:00
|
|
|
}
|
|
|
|
|
2009-03-11 21:03:09 +00:00
|
|
|
Graphics::PixelFormat Palette::colorFormat() const {
|
|
|
|
return _format;
|
|
|
|
}
|
|
|
|
|
2009-03-11 21:11:52 +00:00
|
|
|
void Palette::setColorFormat(const Graphics::PixelFormat format) {
|
|
|
|
_format = format;
|
|
|
|
|
|
|
|
_rBits = (8 - format.rLoss);
|
|
|
|
_gBits = (8 - format.gLoss);
|
|
|
|
_bBits = (8 - format.bLoss);
|
|
|
|
|
|
|
|
_rMax = (1 << _rBits) - 1;
|
|
|
|
_gMax = (1 << _gBits) - 1;
|
|
|
|
_bMax = (1 << _bBits) - 1;
|
|
|
|
}
|
|
|
|
|
2009-03-16 21:30:15 +00:00
|
|
|
void Palette::setEndianType(const EndianType endian) {
|
|
|
|
_bigEndian = isBigEndian(endian);
|
2009-03-16 20:24:28 +00:00
|
|
|
}
|
|
|
|
|
2008-11-20 22:16:18 +00:00
|
|
|
// a.k.a. transformPaletteRange
|
2009-03-16 21:10:15 +00:00
|
|
|
Palette &Palette::saturatedAddColor(Palette& output, byte firstIndex, byte lastIndex, signed r, signed g, signed b) {
|
2009-03-10 20:37:54 +00:00
|
|
|
assert(firstIndex < colorCount() && lastIndex < colorCount());
|
2009-03-16 21:10:15 +00:00
|
|
|
assert(firstIndex < output.colorCount() && lastIndex < output.colorCount());
|
|
|
|
assert(output.colorFormat() == colorFormat());
|
2009-03-10 20:37:54 +00:00
|
|
|
|
2009-03-10 22:09:10 +00:00
|
|
|
for (uint i = firstIndex; i <= lastIndex; i++)
|
2009-03-16 21:10:15 +00:00
|
|
|
output._colors[i] = saturatedAddColor(_colors[i], r, g, b);
|
2009-03-11 21:05:21 +00:00
|
|
|
|
2009-03-16 21:10:15 +00:00
|
|
|
return output;
|
2008-11-20 22:16:18 +00:00
|
|
|
}
|
|
|
|
|
2009-03-24 21:45:35 +00:00
|
|
|
Palette &Palette::saturatedAddNormalizedGray(Palette& output, byte firstIndex, byte lastIndex, double normalizedGray) {
|
|
|
|
// Calculate the gray value rounded up away from zero
|
|
|
|
const double signedHalf = ((normalizedGray >= 0) ? +0.5 : -0.5);
|
|
|
|
const signed r = (signed)(_rMax * normalizedGray + signedHalf);
|
|
|
|
const signed g = (signed)(_gMax * normalizedGray + signedHalf);
|
|
|
|
const signed b = (signed)(_bMax * normalizedGray + signedHalf);
|
|
|
|
|
|
|
|
return saturatedAddColor(output, firstIndex, lastIndex, r, g, b);
|
|
|
|
}
|
|
|
|
|
2008-11-20 22:16:18 +00:00
|
|
|
// a.k.a. transformColor
|
|
|
|
// Parameter color components (i.e. r, g and b) are in range [-7, 7]
|
|
|
|
// e.g. r = 7 sets the resulting color's red component to maximum
|
|
|
|
// e.g. r = -7 sets the resulting color's red component to minimum (i.e. zero)
|
2009-03-16 21:10:15 +00:00
|
|
|
Cine::Palette::Color Palette::saturatedAddColor(Cine::Palette::Color baseColor, signed r, signed g, signed b) const {
|
|
|
|
Cine::Palette::Color result;
|
|
|
|
result.r = CLIP<int>(baseColor.r + r, 0, _rMax);
|
|
|
|
result.g = CLIP<int>(baseColor.g + g, 0, _gMax);
|
|
|
|
result.b = CLIP<int>(baseColor.b + b, 0, _bMax);
|
|
|
|
return result;
|
2009-03-10 20:37:54 +00:00
|
|
|
}
|
2008-11-20 22:16:18 +00:00
|
|
|
|
2009-03-16 21:30:15 +00:00
|
|
|
Palette &Palette::load(const byte *buf, const uint size, const Graphics::PixelFormat format, const uint numColors, const EndianType endian) {
|
2009-03-14 22:36:05 +00:00
|
|
|
assert(format.bytesPerPixel * numColors <= size); // Make sure there's enough input space
|
|
|
|
assert(format.aLoss == 8); // No alpha
|
2009-03-11 20:04:08 +00:00
|
|
|
assert(format.rShift / 8 == (format.rShift + MAX<int>(0, 8 - format.rLoss - 1)) / 8); // R must be inside one byte
|
|
|
|
assert(format.gShift / 8 == (format.gShift + MAX<int>(0, 8 - format.gLoss - 1)) / 8); // G must be inside one byte
|
|
|
|
assert(format.bShift / 8 == (format.bShift + MAX<int>(0, 8 - format.bLoss - 1)) / 8); // B must be inside one byte
|
2009-03-10 20:37:54 +00:00
|
|
|
|
2009-03-16 21:30:15 +00:00
|
|
|
setEndianType(endian);
|
2009-03-11 21:11:52 +00:00
|
|
|
setColorFormat(format);
|
2009-03-10 20:37:54 +00:00
|
|
|
|
|
|
|
_colors.clear();
|
2009-03-10 22:09:10 +00:00
|
|
|
_colors.resize(numColors);
|
2009-03-16 20:24:28 +00:00
|
|
|
|
2009-03-16 21:30:15 +00:00
|
|
|
const int rBytePos = bytePos(format.rShift, format.bytesPerPixel, isBigEndian(endian));
|
|
|
|
const int gBytePos = bytePos(format.gShift, format.bytesPerPixel, isBigEndian(endian));
|
|
|
|
const int bBytePos = bytePos(format.bShift, format.bytesPerPixel, isBigEndian(endian));
|
2009-03-10 20:37:54 +00:00
|
|
|
|
2009-03-10 22:09:10 +00:00
|
|
|
for (uint i = 0; i < numColors; i++) {
|
2009-03-10 20:37:54 +00:00
|
|
|
// _rMax, _gMax, _bMax are also used as masks here
|
2009-03-16 20:24:28 +00:00
|
|
|
_colors[i].r = (buf[i * format.bytesPerPixel + rBytePos] >> (format.rShift % 8)) & _rMax;
|
|
|
|
_colors[i].g = (buf[i * format.bytesPerPixel + gBytePos] >> (format.gShift % 8)) & _gMax;
|
|
|
|
_colors[i].b = (buf[i * format.bytesPerPixel + bBytePos] >> (format.bShift % 8)) & _bMax;
|
2008-11-20 22:16:18 +00:00
|
|
|
}
|
|
|
|
|
2009-03-10 20:37:54 +00:00
|
|
|
return *this;
|
2008-11-20 22:22:47 +00:00
|
|
|
}
|
|
|
|
|
2009-03-16 21:30:15 +00:00
|
|
|
byte *Palette::save(byte *buf, const uint size, const EndianType endian) const {
|
|
|
|
return save(buf, size, colorFormat(), colorCount(), endian);
|
2009-03-14 22:36:05 +00:00
|
|
|
}
|
|
|
|
|
2009-03-16 21:30:15 +00:00
|
|
|
byte *Palette::save(byte *buf, const uint size, const Graphics::PixelFormat format, const EndianType endian) const {
|
|
|
|
return save(buf, size, format, colorCount(), endian);
|
2009-03-14 22:36:05 +00:00
|
|
|
}
|
|
|
|
|
2009-03-16 21:30:15 +00:00
|
|
|
byte *Palette::save(byte *buf, const uint size, const Graphics::PixelFormat format, const uint numColors, const EndianType endian, const byte firstIndex) const {
|
2009-03-14 22:36:05 +00:00
|
|
|
assert(format.bytesPerPixel * numColors <= size); // Make sure there's enough output space
|
|
|
|
assert(format.aLoss == 8); // No alpha
|
|
|
|
assert(format.rShift / 8 == (format.rShift + MAX<int>(0, 8 - format.rLoss - 1)) / 8); // R must be inside one byte
|
|
|
|
assert(format.gShift / 8 == (format.gShift + MAX<int>(0, 8 - format.gLoss - 1)) / 8); // G must be inside one byte
|
|
|
|
assert(format.bShift / 8 == (format.bShift + MAX<int>(0, 8 - format.bLoss - 1)) / 8); // B must be inside one byte
|
2009-03-11 20:44:16 +00:00
|
|
|
|
|
|
|
// Clear the part of the output palette we're going to be writing to with all black
|
2009-03-14 22:36:05 +00:00
|
|
|
memset(buf, 0, format.bytesPerPixel * numColors);
|
|
|
|
|
|
|
|
// Calculate how much bit shifting the color components need (for positioning them correctly)
|
|
|
|
const signed rShiftLeft = (colorFormat().rLoss - (signed) format.rLoss) + (format.rShift % 8);
|
|
|
|
const signed gShiftLeft = (colorFormat().gLoss - (signed) format.gLoss) + (format.gShift % 8);
|
|
|
|
const signed bShiftLeft = (colorFormat().bLoss - (signed) format.bLoss) + (format.bShift % 8);
|
|
|
|
|
|
|
|
// Calculate the byte masks for each color component (for masking away excess bits)
|
|
|
|
const byte rMask = ((1 << (8 - format.rLoss)) - 1) << (format.rShift % 8);
|
|
|
|
const byte gMask = ((1 << (8 - format.gLoss)) - 1) << (format.gShift % 8);
|
|
|
|
const byte bMask = ((1 << (8 - format.bLoss)) - 1) << (format.bShift % 8);
|
2009-03-11 20:44:16 +00:00
|
|
|
|
2009-03-16 21:30:15 +00:00
|
|
|
const int rBytePos = bytePos(format.rShift, format.bytesPerPixel, isBigEndian(endian));
|
|
|
|
const int gBytePos = bytePos(format.gShift, format.bytesPerPixel, isBigEndian(endian));
|
|
|
|
const int bBytePos = bytePos(format.bShift, format.bytesPerPixel, isBigEndian(endian));
|
2009-03-16 20:24:28 +00:00
|
|
|
|
2009-03-11 20:44:16 +00:00
|
|
|
// Save the palette to the output in the specified format
|
2009-03-14 22:36:05 +00:00
|
|
|
for (uint i = firstIndex; i < firstIndex + numColors; i++) {
|
2009-03-11 20:44:16 +00:00
|
|
|
// _rMax, _gMax, _bMax are also used as masks here
|
2009-03-16 20:24:28 +00:00
|
|
|
buf[i * format.bytesPerPixel + rBytePos] |= (shiftByteLeft(_colors[i].r, rShiftLeft) & rMask);
|
|
|
|
buf[i * format.bytesPerPixel + gBytePos] |= (shiftByteLeft(_colors[i].g, gShiftLeft) & gMask);
|
|
|
|
buf[i * format.bytesPerPixel + bBytePos] |= (shiftByteLeft(_colors[i].b, bShiftLeft) & bMask);
|
2009-03-11 20:44:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Return the pointer to the output palette
|
2009-03-14 22:36:05 +00:00
|
|
|
return buf;
|
2009-03-11 21:03:09 +00:00
|
|
|
}
|
|
|
|
|
2006-02-25 00:26:14 +00:00
|
|
|
} // End of namespace Cine
|