2009-08-21 13:52:43 +00:00
|
|
|
/* 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.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2011-01-18 16:18:10 +00:00
|
|
|
#include "graphics/conversion.h"
|
2009-08-21 13:52:43 +00:00
|
|
|
#include "graphics/jpeg.h"
|
2011-01-18 16:18:10 +00:00
|
|
|
#include "graphics/pixelformat.h"
|
2009-08-21 13:52:43 +00:00
|
|
|
|
2011-04-24 08:34:27 +00:00
|
|
|
#include "common/debug.h"
|
2009-08-21 13:52:43 +00:00
|
|
|
#include "common/endian.h"
|
2010-11-19 01:37:04 +00:00
|
|
|
#include "common/stream.h"
|
2011-04-24 08:34:27 +00:00
|
|
|
#include "common/textconsole.h"
|
2009-08-21 13:52:43 +00:00
|
|
|
|
|
|
|
namespace Graphics {
|
|
|
|
|
|
|
|
// Order used to traverse the quantization tables
|
2009-09-01 13:03:12 +00:00
|
|
|
static const uint8 _zigZagOrder[64] = {
|
2009-08-21 13:52:43 +00:00
|
|
|
0, 1, 8, 16, 9, 2, 3, 10,
|
|
|
|
17, 24, 32, 25, 18, 11, 4, 5,
|
|
|
|
12, 19, 26, 33, 40, 48, 41, 34,
|
|
|
|
27, 20, 13, 6, 7, 14, 21, 28,
|
|
|
|
35, 42, 49, 56, 57, 50, 43, 36,
|
|
|
|
29, 22, 15, 23, 30, 37, 44, 51,
|
|
|
|
58, 59, 52, 45, 38, 31, 39, 46,
|
|
|
|
53, 60, 61, 54, 47, 55, 62, 63
|
|
|
|
};
|
|
|
|
|
2011-02-06 13:43:40 +00:00
|
|
|
// IDCT table built with :
|
2011-03-19 10:24:06 +00:00
|
|
|
// _idct8x8[x][y] = cos(((2 * x + 1) * y) * (M_PI / 16.0)) * 0.5;
|
2011-02-06 13:43:40 +00:00
|
|
|
// _idct8x8[x][y] /= sqrt(2.0) if y == 0
|
|
|
|
static const double _idct8x8[8][8] = {
|
|
|
|
{ 0.353553390593274, 0.490392640201615, 0.461939766255643, 0.415734806151273, 0.353553390593274, 0.277785116509801, 0.191341716182545, 0.097545161008064 },
|
|
|
|
{ 0.353553390593274, 0.415734806151273, 0.191341716182545, -0.097545161008064, -0.353553390593274, -0.490392640201615, -0.461939766255643, -0.277785116509801 },
|
|
|
|
{ 0.353553390593274, 0.277785116509801, -0.191341716182545, -0.490392640201615, -0.353553390593274, 0.097545161008064, 0.461939766255643, 0.415734806151273 },
|
|
|
|
{ 0.353553390593274, 0.097545161008064, -0.461939766255643, -0.277785116509801, 0.353553390593274, 0.415734806151273, -0.191341716182545, -0.490392640201615 },
|
|
|
|
{ 0.353553390593274, -0.097545161008064, -0.461939766255643, 0.277785116509801, 0.353553390593274, -0.415734806151273, -0.191341716182545, 0.490392640201615 },
|
|
|
|
{ 0.353553390593274, -0.277785116509801, -0.191341716182545, 0.490392640201615, -0.353553390593273, -0.097545161008064, 0.461939766255643, -0.415734806151273 },
|
|
|
|
{ 0.353553390593274, -0.415734806151273, 0.191341716182545, 0.097545161008064, -0.353553390593274, 0.490392640201615, -0.461939766255643, 0.277785116509801 },
|
|
|
|
{ 0.353553390593274, -0.490392640201615, 0.461939766255643, -0.415734806151273, 0.353553390593273, -0.277785116509801, 0.191341716182545, -0.097545161008064 }
|
2011-01-28 09:46:39 +00:00
|
|
|
};
|
|
|
|
|
2009-08-21 13:52:43 +00:00
|
|
|
JPEG::JPEG() :
|
2011-01-28 09:37:11 +00:00
|
|
|
_stream(NULL), _w(0), _h(0), _numComp(0), _components(NULL), _numScanComp(0),
|
2009-08-21 13:52:43 +00:00
|
|
|
_scanComp(NULL), _currentComp(NULL) {
|
|
|
|
|
|
|
|
// Initialize the quantization tables
|
|
|
|
for (int i = 0; i < JPEG_MAX_QUANT_TABLES; i++)
|
|
|
|
_quant[i] = NULL;
|
|
|
|
|
|
|
|
// Initialize the Huffman tables
|
|
|
|
for (int i = 0; i < 2 * JPEG_MAX_HUFF_TABLES; i++) {
|
|
|
|
_huff[i].count = 0;
|
|
|
|
_huff[i].values = NULL;
|
|
|
|
_huff[i].sizes = NULL;
|
|
|
|
_huff[i].codes = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
JPEG::~JPEG() {
|
|
|
|
reset();
|
|
|
|
}
|
|
|
|
|
2011-01-18 16:18:10 +00:00
|
|
|
Surface *JPEG::getSurface(const PixelFormat &format) {
|
|
|
|
// Make sure we have loaded data
|
|
|
|
if (!isLoaded())
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
// Only accept >8bpp surfaces
|
|
|
|
if (format.bytesPerPixel == 1)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
// Get our component surfaces
|
|
|
|
Graphics::Surface *yComponent = getComponent(1);
|
|
|
|
Graphics::Surface *uComponent = getComponent(2);
|
|
|
|
Graphics::Surface *vComponent = getComponent(3);
|
|
|
|
|
|
|
|
Graphics::Surface *output = new Graphics::Surface();
|
2011-04-17 14:24:52 +00:00
|
|
|
output->create(yComponent->w, yComponent->h, format);
|
2011-01-18 16:18:10 +00:00
|
|
|
|
|
|
|
for (uint16 i = 0; i < output->h; i++) {
|
|
|
|
for (uint16 j = 0; j < output->w; j++) {
|
|
|
|
byte r = 0, g = 0, b = 0;
|
|
|
|
YUV2RGB(*((byte *)yComponent->getBasePtr(j, i)), *((byte *)uComponent->getBasePtr(j, i)), *((byte *)vComponent->getBasePtr(j, i)), r, g, b);
|
|
|
|
if (format.bytesPerPixel == 2)
|
|
|
|
*((uint16 *)output->getBasePtr(j, i)) = format.RGBToColor(r, g, b);
|
|
|
|
else
|
|
|
|
*((uint32 *)output->getBasePtr(j, i)) = format.RGBToColor(r, g, b);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
|
2009-08-21 13:52:43 +00:00
|
|
|
void JPEG::reset() {
|
|
|
|
// Reset member variables
|
2011-01-28 09:37:11 +00:00
|
|
|
_stream = NULL;
|
2009-08-21 13:52:43 +00:00
|
|
|
_w = _h = 0;
|
|
|
|
|
|
|
|
// Free the components
|
|
|
|
for (int c = 0; c < _numComp; c++)
|
|
|
|
_components[c].surface.free();
|
|
|
|
delete[] _components; _components = NULL;
|
|
|
|
_numComp = 0;
|
|
|
|
|
|
|
|
// Free the scan components
|
|
|
|
delete[] _scanComp; _scanComp = NULL;
|
|
|
|
_numScanComp = 0;
|
|
|
|
_currentComp = NULL;
|
|
|
|
|
|
|
|
// Free the quantization tables
|
|
|
|
for (int i = 0; i < JPEG_MAX_QUANT_TABLES; i++) {
|
|
|
|
delete[] _quant[i];
|
|
|
|
_quant[i] = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Free the Huffman tables
|
|
|
|
for (int i = 0; i < 2 * JPEG_MAX_HUFF_TABLES; i++) {
|
|
|
|
_huff[i].count = 0;
|
|
|
|
delete[] _huff[i].values; _huff[i].values = NULL;
|
|
|
|
delete[] _huff[i].sizes; _huff[i].sizes = NULL;
|
|
|
|
delete[] _huff[i].codes; _huff[i].codes = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-01-28 09:37:11 +00:00
|
|
|
bool JPEG::read(Common::SeekableReadStream *stream) {
|
2009-08-21 13:52:43 +00:00
|
|
|
// Reset member variables and tables from previous reads
|
|
|
|
reset();
|
|
|
|
|
|
|
|
// Save the input stream
|
2011-01-28 09:37:11 +00:00
|
|
|
_stream = stream;
|
2009-08-21 13:52:43 +00:00
|
|
|
|
|
|
|
bool ok = true;
|
|
|
|
bool done = false;
|
2011-01-28 09:37:11 +00:00
|
|
|
while (!_stream->eos() && ok && !done) {
|
2009-08-21 13:52:43 +00:00
|
|
|
// Read the marker
|
2009-12-31 10:13:59 +00:00
|
|
|
|
|
|
|
// WORKAROUND: While each and every JPEG file should end with
|
|
|
|
// an EOI (end of image) tag, in reality this may not be the
|
|
|
|
// case. For instance, at least one image in the Masterpiece
|
|
|
|
// edition of Myst doesn't, yet other programs are able to read
|
|
|
|
// the image without complaining.
|
|
|
|
//
|
|
|
|
// Apparently, the customary workaround is to insert a fake
|
|
|
|
// EOI tag.
|
|
|
|
|
2011-01-28 09:37:11 +00:00
|
|
|
uint16 marker = _stream->readByte();
|
2009-12-31 10:13:59 +00:00
|
|
|
bool fakeEOI = false;
|
|
|
|
|
2011-01-28 09:37:11 +00:00
|
|
|
if (_stream->eos()) {
|
2009-12-31 10:13:59 +00:00
|
|
|
fakeEOI = true;
|
|
|
|
marker = 0xFF;
|
|
|
|
}
|
|
|
|
|
2009-08-21 13:52:43 +00:00
|
|
|
if (marker != 0xFF) {
|
|
|
|
error("JPEG: Invalid marker[0]: 0x%02X", marker);
|
|
|
|
ok = false;
|
|
|
|
break;
|
|
|
|
}
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2011-01-28 09:37:11 +00:00
|
|
|
while (marker == 0xFF && !_stream->eos())
|
|
|
|
marker = _stream->readByte();
|
2009-08-21 13:52:43 +00:00
|
|
|
|
2011-01-28 09:37:11 +00:00
|
|
|
if (_stream->eos()) {
|
2009-12-31 10:13:59 +00:00
|
|
|
fakeEOI = true;
|
|
|
|
marker = 0xD9;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fakeEOI)
|
|
|
|
warning("JPEG: Inserted fake EOI");
|
|
|
|
|
2009-08-21 13:52:43 +00:00
|
|
|
// Process the marker data
|
|
|
|
switch (marker) {
|
|
|
|
case 0xC0: // Start Of Frame
|
|
|
|
ok = readSOF0();
|
|
|
|
break;
|
|
|
|
case 0xC4: // Define Huffman Tables
|
|
|
|
ok = readDHT();
|
|
|
|
break;
|
|
|
|
case 0xD8: // Start Of Image
|
|
|
|
break;
|
|
|
|
case 0xD9: // End Of Image
|
|
|
|
done = true;
|
|
|
|
break;
|
|
|
|
case 0xDA: // Start Of Scan
|
|
|
|
ok = readSOS();
|
|
|
|
break;
|
|
|
|
case 0xDB: // Define Quantization Tables
|
|
|
|
ok = readDQT();
|
|
|
|
break;
|
|
|
|
case 0xE0: // JFIF/JFXX segment
|
|
|
|
ok = readJFIF();
|
|
|
|
break;
|
|
|
|
case 0xFE: // Comment
|
2011-01-28 09:37:11 +00:00
|
|
|
_stream->seek(_stream->readUint16BE() - 2, SEEK_CUR);
|
2009-08-21 13:52:43 +00:00
|
|
|
break;
|
|
|
|
default: { // Unknown marker
|
2011-01-28 09:37:11 +00:00
|
|
|
uint16 size = _stream->readUint16BE();
|
2009-08-21 13:52:43 +00:00
|
|
|
warning("JPEG: Unknown marker %02X, skipping %d bytes", marker, size - 2);
|
2011-01-28 09:37:11 +00:00
|
|
|
_stream->seek(size - 2, SEEK_CUR);
|
2009-08-21 13:52:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ok;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool JPEG::readJFIF() {
|
2011-01-28 09:37:11 +00:00
|
|
|
uint16 length = _stream->readUint16BE();
|
|
|
|
uint32 tag = _stream->readUint32BE();
|
2012-01-06 14:51:10 +00:00
|
|
|
if (tag != MKTAG('J', 'F', 'I', 'F')) {
|
2009-12-24 08:26:13 +00:00
|
|
|
warning("JPEG::readJFIF() tag mismatch");
|
|
|
|
return false;
|
|
|
|
}
|
2011-01-28 09:37:11 +00:00
|
|
|
if (_stream->readByte() != 0) { // NULL
|
2009-12-24 08:26:13 +00:00
|
|
|
warning("JPEG::readJFIF() NULL mismatch");
|
2009-08-21 13:52:43 +00:00
|
|
|
return false;
|
2009-12-24 08:26:13 +00:00
|
|
|
}
|
2011-01-28 09:37:11 +00:00
|
|
|
byte majorVersion = _stream->readByte();
|
|
|
|
byte minorVersion = _stream->readByte();
|
2012-01-06 14:51:10 +00:00
|
|
|
if (majorVersion != 1 || minorVersion != 1)
|
2010-09-18 10:55:16 +00:00
|
|
|
warning("JPEG::readJFIF() Non-v1.1 JPEGs may not be handled correctly");
|
2011-01-28 09:37:11 +00:00
|
|
|
/* byte densityUnits = */ _stream->readByte();
|
|
|
|
/* uint16 xDensity = */ _stream->readUint16BE();
|
|
|
|
/* uint16 yDensity = */ _stream->readUint16BE();
|
|
|
|
byte thumbW = _stream->readByte();
|
|
|
|
byte thumbH = _stream->readByte();
|
|
|
|
_stream->seek(thumbW * thumbH * 3, SEEK_CUR); // Ignore thumbnail
|
2009-12-24 08:26:13 +00:00
|
|
|
if (length != (thumbW * thumbH * 3) + 16) {
|
|
|
|
warning("JPEG::readJFIF() length mismatch");
|
|
|
|
return false;
|
|
|
|
}
|
2009-08-21 13:52:43 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Marker 0xC0 (Start Of Frame, Baseline DCT)
|
|
|
|
bool JPEG::readSOF0() {
|
|
|
|
debug(5, "JPEG: readSOF0");
|
2011-01-28 09:37:11 +00:00
|
|
|
uint16 size = _stream->readUint16BE();
|
2009-08-21 13:52:43 +00:00
|
|
|
|
|
|
|
// Read the sample precision
|
2011-01-28 09:37:11 +00:00
|
|
|
uint8 precision = _stream->readByte();
|
2009-08-21 13:52:43 +00:00
|
|
|
if (precision != 8) {
|
|
|
|
warning("JPEG: Just 8 bit precision supported at the moment");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Image size
|
2011-01-28 09:37:11 +00:00
|
|
|
_h = _stream->readUint16BE();
|
|
|
|
_w = _stream->readUint16BE();
|
2009-08-21 13:52:43 +00:00
|
|
|
|
|
|
|
// Number of components
|
2011-01-28 09:37:11 +00:00
|
|
|
_numComp = _stream->readByte();
|
2009-08-21 13:52:43 +00:00
|
|
|
if (size != 8 + 3 * _numComp) {
|
|
|
|
warning("JPEG: Invalid number of components");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Allocate the new components
|
|
|
|
delete[] _components;
|
|
|
|
_components = new Component[_numComp];
|
|
|
|
|
|
|
|
// Read the components details
|
|
|
|
for (int c = 0; c < _numComp; c++) {
|
2011-01-28 09:37:11 +00:00
|
|
|
_components[c].id = _stream->readByte();
|
|
|
|
_components[c].factorH = _stream->readByte();
|
2009-08-21 13:52:43 +00:00
|
|
|
_components[c].factorV = _components[c].factorH & 0xF;
|
|
|
|
_components[c].factorH >>= 4;
|
2011-01-28 09:37:11 +00:00
|
|
|
_components[c].quantTableSelector = _stream->readByte();
|
2009-08-21 13:52:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Marker 0xC4 (Define Huffman Tables)
|
|
|
|
bool JPEG::readDHT() {
|
|
|
|
debug(5, "JPEG: readDHT");
|
2011-01-28 09:37:11 +00:00
|
|
|
uint16 size = _stream->readUint16BE() - 2;
|
|
|
|
uint32 pos = _stream->pos();
|
2009-08-21 13:52:43 +00:00
|
|
|
|
2011-01-28 09:37:11 +00:00
|
|
|
while ((uint32)_stream->pos() < (size + pos)) {
|
2009-08-21 13:52:43 +00:00
|
|
|
// Read the table type and id
|
2011-01-28 09:37:11 +00:00
|
|
|
uint8 tableId = _stream->readByte();
|
2009-08-21 13:52:43 +00:00
|
|
|
uint8 tableType = tableId >> 4; // type 0: DC, 1: AC
|
|
|
|
tableId &= 0xF;
|
|
|
|
uint8 tableNum = (tableId << 1) + tableType;
|
|
|
|
|
|
|
|
// Free the Huffman table
|
|
|
|
delete[] _huff[tableNum].values; _huff[tableNum].values = NULL;
|
|
|
|
delete[] _huff[tableNum].sizes; _huff[tableNum].sizes = NULL;
|
2012-01-06 14:51:10 +00:00
|
|
|
delete[] _huff[tableNum].codes; _huff[tableNum].codes = NULL;
|
2009-08-21 13:52:43 +00:00
|
|
|
|
|
|
|
// Read the number of values for each length
|
|
|
|
uint8 numValues[16];
|
|
|
|
_huff[tableNum].count = 0;
|
|
|
|
for (int len = 0; len < 16; len++) {
|
2011-01-28 09:37:11 +00:00
|
|
|
numValues[len] = _stream->readByte();
|
2009-08-21 13:52:43 +00:00
|
|
|
_huff[tableNum].count += numValues[len];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Allocate memory for the current table
|
|
|
|
_huff[tableNum].values = new uint8[_huff[tableNum].count];
|
|
|
|
_huff[tableNum].sizes = new uint8[_huff[tableNum].count];
|
|
|
|
_huff[tableNum].codes = new uint16[_huff[tableNum].count];
|
|
|
|
|
|
|
|
// Read the table contents
|
|
|
|
int cur = 0;
|
|
|
|
for (int len = 0; len < 16; len++) {
|
|
|
|
for (int i = 0; i < numValues[len]; i++) {
|
2011-01-28 09:37:11 +00:00
|
|
|
_huff[tableNum].values[cur] = _stream->readByte();
|
2009-08-21 13:52:43 +00:00
|
|
|
_huff[tableNum].sizes[cur] = len + 1;
|
|
|
|
cur++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fill the table of Huffman codes
|
|
|
|
cur = 0;
|
|
|
|
uint16 curCode = 0;
|
|
|
|
uint8 curCodeSize = _huff[tableNum].sizes[0];
|
|
|
|
while (cur < _huff[tableNum].count) {
|
|
|
|
// Increase the code size to fit the request
|
|
|
|
while (_huff[tableNum].sizes[cur] != curCodeSize) {
|
|
|
|
curCode <<= 1;
|
|
|
|
curCodeSize++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assign the current code
|
|
|
|
_huff[tableNum].codes[cur] = curCode;
|
|
|
|
curCode++;
|
|
|
|
cur++;
|
|
|
|
}
|
|
|
|
}
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2009-08-21 13:52:43 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Marker 0xDA (Start Of Scan)
|
|
|
|
bool JPEG::readSOS() {
|
|
|
|
debug(5, "JPEG: readSOS");
|
2011-01-28 09:37:11 +00:00
|
|
|
uint16 size = _stream->readUint16BE();
|
2009-08-21 13:52:43 +00:00
|
|
|
|
|
|
|
// Number of scan components
|
2011-01-28 09:37:11 +00:00
|
|
|
_numScanComp = _stream->readByte();
|
2009-08-21 13:52:43 +00:00
|
|
|
if (size != 6 + 2 * _numScanComp) {
|
|
|
|
warning("JPEG: Invalid number of components");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Allocate the new scan components
|
|
|
|
delete[] _scanComp;
|
|
|
|
_scanComp = new Component *[_numScanComp];
|
|
|
|
|
|
|
|
// Reset the maximum sampling factors
|
|
|
|
_maxFactorV = 0;
|
|
|
|
_maxFactorH = 0;
|
|
|
|
|
|
|
|
// Component-specification parameters
|
|
|
|
for (int c = 0; c < _numScanComp; c++) {
|
|
|
|
// Read the desired component id
|
2011-01-28 09:37:11 +00:00
|
|
|
uint8 id = _stream->readByte();
|
2009-08-21 13:52:43 +00:00
|
|
|
|
|
|
|
// Search the component with the specified id
|
|
|
|
bool found = false;
|
|
|
|
for (int i = 0; !found && i < _numComp; i++) {
|
|
|
|
if (_components[i].id == id) {
|
|
|
|
// We found the desired component
|
|
|
|
found = true;
|
|
|
|
|
|
|
|
// Assign the found component to the c'th scan component
|
|
|
|
_scanComp[c] = &_components[i];
|
|
|
|
}
|
|
|
|
}
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2009-08-21 13:52:43 +00:00
|
|
|
if (!found) {
|
|
|
|
warning("JPEG: Invalid component");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read the entropy table selectors
|
2011-01-28 09:37:11 +00:00
|
|
|
_scanComp[c]->DCentropyTableSelector = _stream->readByte();
|
2009-08-21 13:52:43 +00:00
|
|
|
_scanComp[c]->ACentropyTableSelector = _scanComp[c]->DCentropyTableSelector & 0xF;
|
|
|
|
_scanComp[c]->DCentropyTableSelector >>= 4;
|
|
|
|
|
|
|
|
// Calculate the maximum sampling factors
|
|
|
|
if (_scanComp[c]->factorV > _maxFactorV)
|
|
|
|
_maxFactorV = _scanComp[c]->factorV;
|
|
|
|
|
|
|
|
if (_scanComp[c]->factorH > _maxFactorH)
|
|
|
|
_maxFactorH = _scanComp[c]->factorH;
|
|
|
|
|
|
|
|
// Initialize the DC predictor
|
|
|
|
_scanComp[c]->DCpredictor = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Start of spectral selection
|
2011-01-28 09:37:11 +00:00
|
|
|
if (_stream->readByte() != 0) {
|
2009-08-21 13:52:43 +00:00
|
|
|
warning("JPEG: Progressive scanning not supported");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// End of spectral selection
|
2011-01-28 09:37:11 +00:00
|
|
|
if (_stream->readByte() != 63) {
|
2009-08-21 13:52:43 +00:00
|
|
|
warning("JPEG: Progressive scanning not supported");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Successive approximation parameters
|
2011-01-28 09:37:11 +00:00
|
|
|
if (_stream->readByte() != 0) {
|
2009-08-21 13:52:43 +00:00
|
|
|
warning("JPEG: Progressive scanning not supported");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Entropy coded sequence starts, initialize Huffman decoder
|
|
|
|
_bitsNumber = 0;
|
|
|
|
|
|
|
|
// Read all the scan MCUs
|
|
|
|
uint16 xMCU = _w / (_maxFactorH * 8);
|
|
|
|
uint16 yMCU = _h / (_maxFactorV * 8);
|
|
|
|
|
|
|
|
// Check for non- multiple-of-8 dimensions
|
2009-12-24 08:26:13 +00:00
|
|
|
if (_w % (_maxFactorH * 8) != 0)
|
2009-08-21 13:52:43 +00:00
|
|
|
xMCU++;
|
2009-12-24 08:26:13 +00:00
|
|
|
if (_h % (_maxFactorV * 8) != 0)
|
2009-08-21 13:52:43 +00:00
|
|
|
yMCU++;
|
|
|
|
|
2009-12-24 08:26:13 +00:00
|
|
|
// Initialize the scan surfaces
|
|
|
|
for (uint16 c = 0; c < _numScanComp; c++) {
|
2011-04-17 14:24:52 +00:00
|
|
|
_scanComp[c]->surface.create(xMCU * _maxFactorH * 8, yMCU * _maxFactorV * 8, PixelFormat::createFormatCLUT8());
|
2009-12-24 08:26:13 +00:00
|
|
|
}
|
|
|
|
|
2009-08-21 13:52:43 +00:00
|
|
|
bool ok = true;
|
2009-12-24 08:26:13 +00:00
|
|
|
for (int y = 0; ok && (y < yMCU); y++)
|
2009-08-21 13:52:43 +00:00
|
|
|
for (int x = 0; ok && (x < xMCU); x++)
|
|
|
|
ok = readMCU(x, y);
|
|
|
|
|
2009-12-24 08:26:13 +00:00
|
|
|
// Trim Component surfaces back to image height and width
|
|
|
|
// Note: Code using jpeg must use surface.pitch correctly...
|
|
|
|
for (uint16 c = 0; c < _numScanComp; c++) {
|
|
|
|
_scanComp[c]->surface.w = _w;
|
|
|
|
_scanComp[c]->surface.h = _h;
|
|
|
|
}
|
|
|
|
|
2009-08-21 13:52:43 +00:00
|
|
|
return ok;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Marker 0xDB (Define Quantization Tables)
|
|
|
|
bool JPEG::readDQT() {
|
|
|
|
debug(5, "JPEG: readDQT");
|
2011-01-28 09:37:11 +00:00
|
|
|
uint16 size = _stream->readUint16BE() - 2;
|
|
|
|
uint32 pos = _stream->pos();
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2011-01-28 09:37:11 +00:00
|
|
|
while ((uint32)_stream->pos() < (pos + size)) {
|
2009-08-21 13:52:43 +00:00
|
|
|
// Read the table precision and id
|
2011-01-28 09:37:11 +00:00
|
|
|
uint8 tableId = _stream->readByte();
|
2009-08-21 13:52:43 +00:00
|
|
|
bool highPrecision = (tableId & 0xF0) != 0;
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2009-08-21 13:52:43 +00:00
|
|
|
// Validate the table id
|
|
|
|
tableId &= 0xF;
|
|
|
|
if (tableId > JPEG_MAX_QUANT_TABLES) {
|
|
|
|
warning("JPEG: Invalid number of components");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create the new table if necessary
|
|
|
|
if (!_quant[tableId])
|
|
|
|
_quant[tableId] = new uint16[64];
|
|
|
|
|
|
|
|
// Read the table (stored in Zig-Zag order)
|
|
|
|
for (int i = 0; i < 64; i++)
|
2011-01-28 09:37:11 +00:00
|
|
|
_quant[tableId][i] = highPrecision ? _stream->readUint16BE() : _stream->readByte();
|
2009-08-21 13:52:43 +00:00
|
|
|
}
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2009-08-21 13:52:43 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool JPEG::readMCU(uint16 xMCU, uint16 yMCU) {
|
|
|
|
bool ok = true;
|
|
|
|
for (int c = 0; ok && (c < _numComp); c++) {
|
|
|
|
// Set the current component
|
|
|
|
_currentComp = _scanComp[c];
|
|
|
|
|
|
|
|
// Read the data units of the current component
|
|
|
|
for (int y = 0; ok && (y < _scanComp[c]->factorV); y++)
|
|
|
|
for (int x = 0; ok && (x < _scanComp[c]->factorH); x++)
|
|
|
|
ok = readDataUnit(xMCU * _scanComp[c]->factorH + x, yMCU * _scanComp[c]->factorV + y);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ok;
|
|
|
|
}
|
|
|
|
|
2011-02-06 13:43:40 +00:00
|
|
|
void JPEG::idct8x8(float result[64], const int16 dct[64]) {
|
|
|
|
float tmp[64];
|
|
|
|
|
|
|
|
// Apply 1D IDCT to rows
|
|
|
|
for (int y = 0; y < 8; y++) {
|
|
|
|
for (int x = 0; x < 8; x++) {
|
|
|
|
tmp[y + x * 8] = dct[0] * _idct8x8[x][0]
|
2012-01-06 14:51:10 +00:00
|
|
|
+ dct[1] * _idct8x8[x][1]
|
|
|
|
+ dct[2] * _idct8x8[x][2]
|
|
|
|
+ dct[3] * _idct8x8[x][3]
|
|
|
|
+ dct[4] * _idct8x8[x][4]
|
|
|
|
+ dct[5] * _idct8x8[x][5]
|
|
|
|
+ dct[6] * _idct8x8[x][6]
|
|
|
|
+ dct[7] * _idct8x8[x][7];
|
2011-02-06 13:43:40 +00:00
|
|
|
}
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2011-02-06 13:43:40 +00:00
|
|
|
dct += 8;
|
|
|
|
}
|
2009-08-21 13:52:43 +00:00
|
|
|
|
2011-02-06 13:43:40 +00:00
|
|
|
// Apply 1D IDCT to columns
|
|
|
|
for (int x = 0; x < 8; x++) {
|
|
|
|
const float *u = tmp + x * 8;
|
|
|
|
for (int y = 0; y < 8; y++) {
|
|
|
|
result[y * 8 + x] = u[0] * _idct8x8[y][0]
|
2012-01-06 14:51:10 +00:00
|
|
|
+ u[1] * _idct8x8[y][1]
|
|
|
|
+ u[2] * _idct8x8[y][2]
|
|
|
|
+ u[3] * _idct8x8[y][3]
|
|
|
|
+ u[4] * _idct8x8[y][4]
|
|
|
|
+ u[5] * _idct8x8[y][5]
|
|
|
|
+ u[6] * _idct8x8[y][6]
|
|
|
|
+ u[7] * _idct8x8[y][7];
|
2011-02-06 13:43:40 +00:00
|
|
|
}
|
|
|
|
}
|
2009-08-21 13:52:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool JPEG::readDataUnit(uint16 x, uint16 y) {
|
|
|
|
// Prepare an empty data array
|
|
|
|
int16 readData[64];
|
|
|
|
for (int i = 1; i < 64; i++)
|
|
|
|
readData[i] = 0;
|
|
|
|
|
|
|
|
// Read the DC component
|
|
|
|
readData[0] = _currentComp->DCpredictor + readDC();
|
|
|
|
_currentComp->DCpredictor = readData[0];
|
|
|
|
|
|
|
|
// Read the AC components (stored in Zig-Zag)
|
|
|
|
readAC(readData);
|
|
|
|
|
|
|
|
// Calculate the DCT coefficients from the input sequence
|
|
|
|
int16 DCT[64];
|
2009-12-24 08:26:13 +00:00
|
|
|
for (uint8 i = 0; i < 64; i++) {
|
2009-08-21 13:52:43 +00:00
|
|
|
// Dequantize
|
|
|
|
int16 val = readData[i];
|
|
|
|
int16 quant = _quant[_currentComp->quantTableSelector][i];
|
|
|
|
val *= quant;
|
|
|
|
|
|
|
|
// Store the normalized coefficients, undoing the Zig-Zag
|
|
|
|
DCT[_zigZagOrder[i]] = val;
|
|
|
|
}
|
|
|
|
|
2011-02-06 13:43:40 +00:00
|
|
|
// Apply the IDCT
|
2009-08-21 13:52:43 +00:00
|
|
|
float result[64];
|
2011-02-06 13:43:40 +00:00
|
|
|
idct8x8(result, DCT);
|
2009-08-21 13:52:43 +00:00
|
|
|
|
|
|
|
// Level shift to make the values unsigned
|
|
|
|
for (int i = 0; i < 64; i++) {
|
2011-02-06 13:43:40 +00:00
|
|
|
result[i] = result[i] + 128;
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2009-08-21 13:52:43 +00:00
|
|
|
if (result[i] < 0)
|
|
|
|
result[i] = 0;
|
|
|
|
|
|
|
|
if (result[i] > 255)
|
|
|
|
result[i] = 255;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Paint the component surface
|
|
|
|
uint8 scalingV = _maxFactorV / _currentComp->factorV;
|
|
|
|
uint8 scalingH = _maxFactorH / _currentComp->factorH;
|
|
|
|
|
|
|
|
// Convert coordinates from MCU blocks to pixels
|
|
|
|
x <<= 3;
|
|
|
|
y <<= 3;
|
|
|
|
|
2009-12-24 08:26:13 +00:00
|
|
|
for (uint8 j = 0; j < 8; j++) {
|
|
|
|
for (uint16 sV = 0; sV < scalingV; sV++) {
|
2009-08-21 13:52:43 +00:00
|
|
|
// Get the beginning of the block line
|
|
|
|
byte *ptr = (byte *)_currentComp->surface.getBasePtr(x * scalingH, (y + j) * scalingV + sV);
|
|
|
|
|
2009-12-24 08:26:13 +00:00
|
|
|
for (uint8 i = 0; i < 8; i++) {
|
|
|
|
for (uint16 sH = 0; sH < scalingH; sH++) {
|
2009-08-21 13:52:43 +00:00
|
|
|
*ptr = (byte)(result[j * 8 + i]);
|
|
|
|
ptr++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
int16 JPEG::readDC() {
|
|
|
|
// DC is type 0
|
|
|
|
uint8 tableNum = _currentComp->DCentropyTableSelector << 1;
|
|
|
|
|
|
|
|
// Get the number of bits to read
|
|
|
|
uint8 numBits = readHuff(tableNum);
|
|
|
|
|
|
|
|
// Read the requested bits
|
|
|
|
return readSignedBits(numBits);
|
|
|
|
}
|
|
|
|
|
|
|
|
void JPEG::readAC(int16 *out) {
|
|
|
|
// AC is type 1
|
|
|
|
uint8 tableNum = (_currentComp->ACentropyTableSelector << 1) + 1;
|
|
|
|
|
|
|
|
// Start reading AC element 1
|
|
|
|
uint8 cur = 1;
|
|
|
|
while (cur < 64) {
|
|
|
|
uint8 s = readHuff(tableNum);
|
|
|
|
uint8 r = s >> 4;
|
|
|
|
s &= 0xF;
|
|
|
|
|
|
|
|
if (s == 0) {
|
|
|
|
if (r == 15) {
|
|
|
|
// Skip 16 values
|
|
|
|
cur += 16;
|
|
|
|
} else {
|
|
|
|
// EOB: end of block
|
|
|
|
cur = 64;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Skip r values
|
|
|
|
cur += r;
|
|
|
|
|
|
|
|
// Read the next value
|
|
|
|
out[cur] = readSignedBits(s);
|
|
|
|
cur++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int16 JPEG::readSignedBits(uint8 numBits) {
|
|
|
|
uint16 ret = 0;
|
|
|
|
if (numBits > 16) error("requested %d bits", numBits); //XXX
|
|
|
|
|
|
|
|
// MSB=0 for negatives, 1 for positives
|
|
|
|
for (int i = 0; i < numBits; i++)
|
|
|
|
ret = (ret << 1) + readBit();
|
|
|
|
|
|
|
|
// Extend sign bits (PAG109)
|
2012-01-06 14:51:10 +00:00
|
|
|
if (!(ret >> (numBits - 1))) {
|
|
|
|
uint16 tmp = ((uint16) - 1 << numBits) + 1;
|
2009-08-21 13:52:43 +00:00
|
|
|
ret = ret + tmp;
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: optimize?
|
|
|
|
uint8 JPEG::readHuff(uint8 table) {
|
|
|
|
bool foundCode = false;
|
|
|
|
uint8 val = 0;
|
|
|
|
|
|
|
|
uint8 cur = 0;
|
|
|
|
uint8 codeSize = 1;
|
|
|
|
uint16 code = readBit();
|
|
|
|
while (!foundCode) {
|
|
|
|
// Prepare a code of the current size
|
|
|
|
while (codeSize < _huff[table].sizes[cur]) {
|
|
|
|
code = (code << 1) + readBit();
|
|
|
|
codeSize++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Compare the codes of the current size
|
|
|
|
while (!foundCode && (codeSize == _huff[table].sizes[cur])) {
|
|
|
|
if (code == _huff[table].codes[cur]) {
|
|
|
|
// Found the code
|
|
|
|
val = _huff[table].values[cur];
|
|
|
|
foundCode = true;
|
|
|
|
} else {
|
|
|
|
// Continue reading
|
|
|
|
cur++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8 JPEG::readBit() {
|
|
|
|
// Read a whole byte if necessary
|
|
|
|
if (_bitsNumber == 0) {
|
2011-01-28 09:37:11 +00:00
|
|
|
_bitsData = _stream->readByte();
|
2009-08-21 13:52:43 +00:00
|
|
|
_bitsNumber = 8;
|
|
|
|
|
|
|
|
// Detect markers
|
|
|
|
if (_bitsData == 0xFF) {
|
2011-01-28 09:37:11 +00:00
|
|
|
uint8 byte2 = _stream->readByte();
|
2009-08-21 13:52:43 +00:00
|
|
|
|
|
|
|
// A stuffed 0 validates the previous byte
|
|
|
|
if (byte2 != 0) {
|
|
|
|
if (byte2 == 0xDC) {
|
|
|
|
// DNL marker: Define Number of Lines
|
|
|
|
// TODO: terminate scan
|
2011-05-02 11:40:13 +00:00
|
|
|
warning("DNL marker detected: terminate scan");
|
2009-08-21 13:52:43 +00:00
|
|
|
} else {
|
2011-05-02 11:40:13 +00:00
|
|
|
warning("Error: marker 0x%02X read in entropy data", byte2);
|
2009-08-21 13:52:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_bitsNumber--;
|
|
|
|
|
|
|
|
return (_bitsData & (1 << _bitsNumber)) ? 1 : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
Surface *JPEG::getComponent(uint c) {
|
|
|
|
for (int i = 0; i < _numComp; i++)
|
|
|
|
if (_components[i].id == c) // We found the desired component
|
|
|
|
return &_components[i].surface;
|
|
|
|
|
2011-01-28 09:50:05 +00:00
|
|
|
error("JPEG::getComponent: No component %d present", c);
|
2009-08-21 13:52:43 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // End of Graphics namespace
|