175 lines
4.5 KiB
C++
175 lines
4.5 KiB
C++
/*
|
|
* @file llimagepng.cpp
|
|
* @brief LLImageFormatted glue to encode / decode PNG files.
|
|
*
|
|
* $LicenseInfo:firstyear=2007&license=viewerlgpl$
|
|
* Second Life Viewer Source Code
|
|
* Copyright (C) 2010, Linden Research, Inc.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation;
|
|
* version 2.1 of the License only.
|
|
*
|
|
* This library 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*
|
|
* Linden Research, Inc., 945 Battery Street, San Francisco, CA 94111 USA
|
|
* $/LicenseInfo$
|
|
*/
|
|
|
|
#include "linden_common.h"
|
|
#include "stdtypes.h"
|
|
#include "llerror.h"
|
|
|
|
#include "llimage.h"
|
|
#include "llpngwrapper.h"
|
|
#include "llimagepng.h"
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// LLImagePNG
|
|
// ---------------------------------------------------------------------------
|
|
LLImagePNG::LLImagePNG()
|
|
: LLImageFormatted(IMG_CODEC_PNG)
|
|
{
|
|
}
|
|
|
|
LLImagePNG::~LLImagePNG()
|
|
{
|
|
}
|
|
|
|
// Virtual
|
|
// Parse PNG image information and set the appropriate
|
|
// width, height and component (channel) information.
|
|
BOOL LLImagePNG::updateData()
|
|
{
|
|
resetLastError();
|
|
|
|
// Check to make sure that this instance has been initialized with data
|
|
if (!getData() || (0 == getDataSize()))
|
|
{
|
|
setLastError("Uninitialized instance of LLImagePNG");
|
|
return FALSE;
|
|
}
|
|
|
|
// Decode the PNG data and extract sizing information
|
|
LLPngWrapper pngWrapper;
|
|
if (!pngWrapper.isValidPng(getData()))
|
|
{
|
|
setLastError("LLImagePNG data does not have a valid PNG header!");
|
|
return FALSE;
|
|
}
|
|
|
|
LLPngWrapper::ImageInfo infop;
|
|
if (! pngWrapper.readPng(getData(), NULL, &infop))
|
|
{
|
|
setLastError(pngWrapper.getErrorMessage());
|
|
return FALSE;
|
|
}
|
|
|
|
setSize(infop.mWidth, infop.mHeight, infop.mComponents);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
// Virtual
|
|
// Decode an in-memory PNG image into the raw RGB or RGBA format
|
|
// used within SecondLife.
|
|
BOOL LLImagePNG::decode(LLImageRaw* raw_image, F32 decode_time)
|
|
{
|
|
llassert_always(raw_image);
|
|
|
|
resetLastError();
|
|
|
|
// Check to make sure that this instance has been initialized with data
|
|
if (!getData() || (0 == getDataSize()))
|
|
{
|
|
setLastError("LLImagePNG trying to decode an image with no data!");
|
|
return FALSE;
|
|
}
|
|
|
|
// Decode the PNG data into the raw image
|
|
LLPngWrapper pngWrapper;
|
|
if (!pngWrapper.isValidPng(getData()))
|
|
{
|
|
setLastError("LLImagePNG data does not have a valid PNG header!");
|
|
return FALSE;
|
|
}
|
|
|
|
if (! pngWrapper.readPng(getData(), raw_image))
|
|
{
|
|
setLastError(pngWrapper.getErrorMessage());
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
// Virtual
|
|
// Encode the in memory RGB image into PNG format.
|
|
BOOL LLImagePNG::encode(const LLImageRaw* raw_image, F32 encode_time)
|
|
{
|
|
llassert_always(raw_image);
|
|
|
|
resetLastError();
|
|
|
|
// Image logical size
|
|
setSize(raw_image->getWidth(), raw_image->getHeight(), raw_image->getComponents());
|
|
|
|
U32 bufferSize = getWidth() * getHeight() * getComponents() + 1024;
|
|
|
|
//New implementation
|
|
allocateData(bufferSize); //Set to largest possible size.
|
|
if(isBufferInvalid()) //Checking
|
|
{
|
|
setLastError("LLImagePNG::encode failed allocateData");
|
|
return FALSE;
|
|
}
|
|
|
|
// Delegate actual encoding work to wrapper
|
|
LLPngWrapper pngWrapper;
|
|
if (! pngWrapper.writePng(raw_image, getData()))
|
|
{
|
|
setLastError(pngWrapper.getErrorMessage());
|
|
deleteData();
|
|
return FALSE;
|
|
}
|
|
|
|
// Resize internal buffer.
|
|
if(!reallocateData(pngWrapper.getFinalSize())) //Shrink. Returns NULL on failure.
|
|
{
|
|
setLastError("LLImagePNG::encode failed reallocateData");
|
|
deleteData();
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
|
|
|
|
/*U8* mTmpWriteBuffer = new U8[ bufferSize ];
|
|
|
|
// Delegate actual encoding work to wrapper
|
|
LLPngWrapper pngWrapper;
|
|
if (! pngWrapper.writePng(raw_image, mTmpWriteBuffer))
|
|
{
|
|
setLastError(pngWrapper.getErrorMessage());
|
|
delete[] mTmpWriteBuffer;
|
|
return FALSE;
|
|
}
|
|
|
|
// Resize internal buffer and copy from temp
|
|
U32 encodedSize = pngWrapper.getFinalSize();
|
|
allocateData(encodedSize);
|
|
memcpy(getData(), mTmpWriteBuffer, encodedSize);
|
|
|
|
delete[] mTmpWriteBuffer;
|
|
|
|
return TRUE;*/
|
|
}
|
|
|