mirror of
https://github.com/ZDoom/raze-gles.git
synced 2024-12-26 11:40:44 +00:00
340 lines
9.1 KiB
C++
340 lines
9.1 KiB
C++
|
/*
|
||
|
** jpegtexture.cpp
|
||
|
** Texture class for JPEG images
|
||
|
**
|
||
|
**---------------------------------------------------------------------------
|
||
|
** Copyright 2006-2007 Randy Heit
|
||
|
** All rights reserved.
|
||
|
**
|
||
|
** Redistribution and use in source and binary forms, with or without
|
||
|
** modification, are permitted provided that the following conditions
|
||
|
** are met:
|
||
|
**
|
||
|
** 1. Redistributions of source code must retain the above copyright
|
||
|
** notice, this list of conditions and the following disclaimer.
|
||
|
** 2. Redistributions in binary form must reproduce the above copyright
|
||
|
** notice, this list of conditions and the following disclaimer in the
|
||
|
** documentation and/or other materials provided with the distribution.
|
||
|
** 3. The name of the author may not be used to endorse or promote products
|
||
|
** derived from this software without specific prior written permission.
|
||
|
**
|
||
|
** THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
||
|
** IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
||
|
** OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
||
|
** IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||
|
** INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
||
|
** NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||
|
** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||
|
** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||
|
** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
||
|
** THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||
|
**---------------------------------------------------------------------------
|
||
|
**
|
||
|
**
|
||
|
*/
|
||
|
|
||
|
#include <stdio.h>
|
||
|
|
||
|
#include "files.h"
|
||
|
#include "printf.h"
|
||
|
#include "bitmap.h"
|
||
|
#include "image.h"
|
||
|
#include "cache1d.h"
|
||
|
|
||
|
extern "C"
|
||
|
{
|
||
|
#include <jpeglib.h>
|
||
|
}
|
||
|
|
||
|
|
||
|
struct FLumpSourceMgr : public jpeg_source_mgr
|
||
|
{
|
||
|
FileReader *Lump;
|
||
|
JOCTET Buffer[4096];
|
||
|
bool StartOfFile;
|
||
|
|
||
|
FLumpSourceMgr (FileReader *lump, j_decompress_ptr cinfo);
|
||
|
static void InitSource (j_decompress_ptr cinfo);
|
||
|
static boolean FillInputBuffer (j_decompress_ptr cinfo);
|
||
|
static void SkipInputData (j_decompress_ptr cinfo, long num_bytes);
|
||
|
static void TermSource (j_decompress_ptr cinfo);
|
||
|
};
|
||
|
|
||
|
|
||
|
//==========================================================================
|
||
|
//
|
||
|
//
|
||
|
//
|
||
|
//==========================================================================
|
||
|
|
||
|
void FLumpSourceMgr::InitSource (j_decompress_ptr cinfo)
|
||
|
{
|
||
|
((FLumpSourceMgr *)(cinfo->src))->StartOfFile = true;
|
||
|
}
|
||
|
|
||
|
//==========================================================================
|
||
|
//
|
||
|
//
|
||
|
//
|
||
|
//==========================================================================
|
||
|
|
||
|
boolean FLumpSourceMgr::FillInputBuffer (j_decompress_ptr cinfo)
|
||
|
{
|
||
|
FLumpSourceMgr *me = (FLumpSourceMgr *)(cinfo->src);
|
||
|
auto nbytes = me->Lump->Read (me->Buffer, sizeof(me->Buffer));
|
||
|
|
||
|
if (nbytes <= 0)
|
||
|
{
|
||
|
me->Buffer[0] = (JOCTET)0xFF;
|
||
|
me->Buffer[1] = (JOCTET)JPEG_EOI;
|
||
|
nbytes = 2;
|
||
|
}
|
||
|
me->next_input_byte = me->Buffer;
|
||
|
me->bytes_in_buffer = nbytes;
|
||
|
me->StartOfFile = false;
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
//==========================================================================
|
||
|
//
|
||
|
//
|
||
|
//
|
||
|
//==========================================================================
|
||
|
|
||
|
void FLumpSourceMgr::SkipInputData (j_decompress_ptr cinfo, long num_bytes)
|
||
|
{
|
||
|
FLumpSourceMgr *me = (FLumpSourceMgr *)(cinfo->src);
|
||
|
if (num_bytes <= (long)me->bytes_in_buffer)
|
||
|
{
|
||
|
me->bytes_in_buffer -= num_bytes;
|
||
|
me->next_input_byte += num_bytes;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
num_bytes -= (long)me->bytes_in_buffer;
|
||
|
me->Lump->Seek (num_bytes, FileReader::SeekCur);
|
||
|
FillInputBuffer (cinfo);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//==========================================================================
|
||
|
//
|
||
|
//
|
||
|
//
|
||
|
//==========================================================================
|
||
|
|
||
|
void FLumpSourceMgr::TermSource (j_decompress_ptr cinfo)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
//==========================================================================
|
||
|
//
|
||
|
//
|
||
|
//
|
||
|
//==========================================================================
|
||
|
|
||
|
FLumpSourceMgr::FLumpSourceMgr (FileReader *lump, j_decompress_ptr cinfo)
|
||
|
: Lump (lump)
|
||
|
{
|
||
|
cinfo->src = this;
|
||
|
init_source = InitSource;
|
||
|
fill_input_buffer = FillInputBuffer;
|
||
|
skip_input_data = SkipInputData;
|
||
|
resync_to_restart = jpeg_resync_to_restart;
|
||
|
term_source = TermSource;
|
||
|
bytes_in_buffer = 0;
|
||
|
next_input_byte = NULL;
|
||
|
}
|
||
|
|
||
|
//==========================================================================
|
||
|
//
|
||
|
//
|
||
|
//
|
||
|
//==========================================================================
|
||
|
|
||
|
void JPEG_ErrorExit (j_common_ptr cinfo)
|
||
|
{
|
||
|
(*cinfo->err->output_message) (cinfo);
|
||
|
throw -1;
|
||
|
}
|
||
|
|
||
|
//==========================================================================
|
||
|
//
|
||
|
//
|
||
|
//
|
||
|
//==========================================================================
|
||
|
|
||
|
void JPEG_OutputMessage (j_common_ptr cinfo)
|
||
|
{
|
||
|
char buffer[JMSG_LENGTH_MAX];
|
||
|
|
||
|
(*cinfo->err->format_message) (cinfo, buffer);
|
||
|
Printf (TEXTCOLOR_ORANGE "JPEG failure: %s\n", buffer);
|
||
|
}
|
||
|
|
||
|
//==========================================================================
|
||
|
//
|
||
|
// A JPEG texture
|
||
|
//
|
||
|
//==========================================================================
|
||
|
|
||
|
class FJPEGTexture : public FImageSource
|
||
|
{
|
||
|
public:
|
||
|
FJPEGTexture (int width, int height);
|
||
|
|
||
|
int CopyPixels(FBitmap *bmp, int conversion) override;
|
||
|
};
|
||
|
|
||
|
//==========================================================================
|
||
|
//
|
||
|
//
|
||
|
//
|
||
|
//==========================================================================
|
||
|
|
||
|
FImageSource *JPEGImage_TryCreate(FileReader & data)
|
||
|
{
|
||
|
union
|
||
|
{
|
||
|
uint32_t dw;
|
||
|
uint16_t w[2];
|
||
|
uint8_t b[4];
|
||
|
} first4bytes;
|
||
|
|
||
|
data.Seek(0, FileReader::SeekSet);
|
||
|
if (data.Read(&first4bytes, 4) < 4) return NULL;
|
||
|
|
||
|
if (first4bytes.b[0] != 0xFF || first4bytes.b[1] != 0xD8 || first4bytes.b[2] != 0xFF)
|
||
|
return NULL;
|
||
|
|
||
|
// Find the SOFn marker to extract the image dimensions,
|
||
|
// where n is 0, 1, or 2 (other types are unsupported).
|
||
|
while ((unsigned)first4bytes.b[3] - 0xC0 >= 3)
|
||
|
{
|
||
|
if (data.Read (first4bytes.w, 2) != 2)
|
||
|
{
|
||
|
return NULL;
|
||
|
}
|
||
|
data.Seek (BigShort(first4bytes.w[0]) - 2, FileReader::SeekCur);
|
||
|
if (data.Read (first4bytes.b + 2, 2) != 2 || first4bytes.b[2] != 0xFF)
|
||
|
{
|
||
|
return NULL;
|
||
|
}
|
||
|
}
|
||
|
if (data.Read (first4bytes.b, 3) != 3)
|
||
|
{
|
||
|
return NULL;
|
||
|
}
|
||
|
if (BigShort (first4bytes.w[0]) < 5)
|
||
|
{
|
||
|
return NULL;
|
||
|
}
|
||
|
if (data.Read (first4bytes.b, 4) != 4)
|
||
|
{
|
||
|
return NULL;
|
||
|
}
|
||
|
return new FJPEGTexture (BigShort(first4bytes.w[1]), BigShort(first4bytes.w[0]));
|
||
|
}
|
||
|
|
||
|
//==========================================================================
|
||
|
//
|
||
|
//
|
||
|
//
|
||
|
//==========================================================================
|
||
|
|
||
|
FJPEGTexture::FJPEGTexture (int width, int height)
|
||
|
{
|
||
|
bMasked = false;
|
||
|
|
||
|
Width = width;
|
||
|
Height = height;
|
||
|
}
|
||
|
|
||
|
//===========================================================================
|
||
|
//
|
||
|
// FJPEGTexture::CopyPixels
|
||
|
//
|
||
|
// Preserves the full color information (unlike software mode)
|
||
|
//
|
||
|
//===========================================================================
|
||
|
|
||
|
int FJPEGTexture::CopyPixels(FBitmap *bmp, int conversion)
|
||
|
{
|
||
|
PalEntry pe[256];
|
||
|
|
||
|
auto lump = kopenFileReader(Name, 0);
|
||
|
if (!lump.isOpen()) return -1; // Just leave the texture blank.
|
||
|
|
||
|
jpeg_decompress_struct cinfo;
|
||
|
jpeg_error_mgr jerr;
|
||
|
|
||
|
cinfo.err = jpeg_std_error(&jerr);
|
||
|
cinfo.err->output_message = JPEG_OutputMessage;
|
||
|
cinfo.err->error_exit = JPEG_ErrorExit;
|
||
|
jpeg_create_decompress(&cinfo);
|
||
|
|
||
|
FLumpSourceMgr sourcemgr(&lump, &cinfo);
|
||
|
try
|
||
|
{
|
||
|
jpeg_read_header(&cinfo, TRUE);
|
||
|
|
||
|
if (!((cinfo.out_color_space == JCS_RGB && cinfo.num_components == 3) ||
|
||
|
(cinfo.out_color_space == JCS_CMYK && cinfo.num_components == 4) ||
|
||
|
(cinfo.out_color_space == JCS_YCbCr && cinfo.num_components == 3) ||
|
||
|
(cinfo.out_color_space == JCS_GRAYSCALE && cinfo.num_components == 1)))
|
||
|
{
|
||
|
Printf(TEXTCOLOR_ORANGE "Unsupported color format in %s\n", Name.GetChars());
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
jpeg_start_decompress(&cinfo);
|
||
|
|
||
|
int yc = 0;
|
||
|
TArray<uint8_t> buff(cinfo.output_height * cinfo.output_width * cinfo.output_components, true);
|
||
|
|
||
|
while (cinfo.output_scanline < cinfo.output_height)
|
||
|
{
|
||
|
uint8_t * ptr = buff.Data() + cinfo.output_width * cinfo.output_components * yc;
|
||
|
jpeg_read_scanlines(&cinfo, &ptr, 1);
|
||
|
yc++;
|
||
|
}
|
||
|
|
||
|
switch (cinfo.out_color_space)
|
||
|
{
|
||
|
case JCS_RGB:
|
||
|
bmp->CopyPixelDataRGB(0, 0, buff.Data(), cinfo.output_width, cinfo.output_height,
|
||
|
3, cinfo.output_width * cinfo.output_components, 0, CF_RGB);
|
||
|
break;
|
||
|
|
||
|
case JCS_GRAYSCALE:
|
||
|
for (int i = 0; i < 256; i++) pe[i] = PalEntry(255, i, i, i); // default to a gray map
|
||
|
bmp->CopyPixelData(0, 0, buff.Data(), cinfo.output_width, cinfo.output_height,
|
||
|
1, cinfo.output_width, 0, pe);
|
||
|
break;
|
||
|
|
||
|
case JCS_CMYK:
|
||
|
bmp->CopyPixelDataRGB(0, 0, buff.Data(), cinfo.output_width, cinfo.output_height,
|
||
|
4, cinfo.output_width * cinfo.output_components, 0, CF_CMYK);
|
||
|
break;
|
||
|
|
||
|
case JCS_YCbCr:
|
||
|
bmp->CopyPixelDataRGB(0, 0, buff.Data(), cinfo.output_width, cinfo.output_height,
|
||
|
4, cinfo.output_width * cinfo.output_components, 0, CF_YCbCr);
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
assert(0);
|
||
|
break;
|
||
|
}
|
||
|
jpeg_finish_decompress(&cinfo);
|
||
|
}
|
||
|
}
|
||
|
catch (int)
|
||
|
{
|
||
|
Printf(TEXTCOLOR_ORANGE "JPEG error in %s\n", Name.GetChars());
|
||
|
}
|
||
|
jpeg_destroy_decompress(&cinfo);
|
||
|
return 0;
|
||
|
}
|
||
|
|