Combine image functionality from PVRTCEncoder into Base library.

This commit is contained in:
Pavel Krajcevski 2013-10-04 18:35:18 -04:00
parent ab598c9ef7
commit 4baf2ce311
24 changed files with 669 additions and 388 deletions

View file

@ -41,29 +41,48 @@
* <http://gamma.cs.unc.edu/FasTC/> * <http://gamma.cs.unc.edu/FasTC/>
*/ */
#ifndef __TEXCOMP_IMAGE_H__ #ifndef FASTC_BASE_INCLUDE_IMAGE_H_
#define __TEXCOMP_IMAGE_H__ #define FASTC_BASE_INCLUDE_IMAGE_H_
#include "TexCompTypes.h" #include "TexCompTypes.h"
#include "ImageFwd.h"
class Image { namespace FasTC {
template<typename U, typename V>
extern double ComputePSNR(Image<U> *img1, Image<V> *img2);
// Forward declare
template<typename PixelType>
class Image {
public: public:
Image(uint32 width, uint32 height);
Image(uint32 width, uint32 height,
const PixelType *pixels,
bool bBlockStreamOrder = false);
Image(uint32 width, uint32 height, Image(uint32 width, uint32 height,
const uint32 *pixels, const uint32 *pixels,
bool bBlockStreamOrder = false); bool bBlockStreamOrder = false);
Image(const Image &); Image(const Image<PixelType> &);
Image &operator=(const Image &); Image &operator=(const Image<PixelType> &);
virtual ~Image(); virtual ~Image();
virtual Image *Clone() const { virtual Image *Clone() const {
return new Image(*this); return new Image(*this);
}; };
const uint8 *RawData() const { return m_Data; } PixelType &operator()(uint32 i, uint32 j);
const PixelType &operator()(uint32 i, uint32 j) const;
// Reads a buffer full of pixels and stores them in the
// data associated with this image.
virtual bool ReadPixels(const uint32 *rgba);
const PixelType *GetPixels() const { return m_Pixels; }
uint32 GetWidth() const { return m_Width; } uint32 GetWidth() const { return m_Width; }
uint32 GetHeight() const { return m_Height; } uint32 GetHeight() const { return m_Height; }
uint32 GetNumPixels() const { return GetWidth() * GetHeight(); }
void SetBlockStreamOrder(bool flag) { void SetBlockStreamOrder(bool flag) {
if(flag) { if(flag) {
@ -74,25 +93,34 @@ class Image {
} }
bool GetBlockStreamOrder() const { return m_bBlockStreamOrder; } bool GetBlockStreamOrder() const { return m_bBlockStreamOrder; }
double ComputePSNR(Image *other); template<typename OtherPixelType>
double ComputePSNR(Image<OtherPixelType> *other) {
virtual void ComputeRGBA() { } return FasTC::ComputePSNR(this, other);
virtual const uint32 *GetRGBA() const {
return reinterpret_cast<const uint32 *>(RawData());
} }
// Function to allow derived classes to populate the pixel array.
// This may involve decompressing a compressed image or otherwise
// processing some data in order to populate the m_Pixels pointer.
// This function should use SetImageData in order to set all of the
// appropriate pixels.
virtual void ComputePixels() { }
private: private:
uint32 m_Width; uint32 m_Width;
uint32 m_Height; uint32 m_Height;
bool m_bBlockStreamOrder; bool m_bBlockStreamOrder;
PixelType *m_Pixels;
protected: protected:
uint32 m_DataSz;
uint8 *m_Data; void SetImageData(uint32 width, uint32 height, PixelType *data);
void ConvertToBlockStreamOrder(); void ConvertToBlockStreamOrder();
void ConvertFromBlockStreamOrder(); void ConvertFromBlockStreamOrder();
}; };
} // namespace FasTC
#endif // __TEXCOMP_IMAGE_H__ #endif // __TEXCOMP_IMAGE_H__

63
Base/include/ImageFwd.h Normal file
View file

@ -0,0 +1,63 @@
/* FasTC
* Copyright (c) 2013 University of North Carolina at Chapel Hill.
* All rights reserved.
*
* Permission to use, copy, modify, and distribute this software and its
* documentation for educational, research, and non-profit purposes, without
* fee, and without a written agreement is hereby granted, provided that the
* above copyright notice, this paragraph, and the following four paragraphs
* appear in all copies.
*
* Permission to incorporate this software into commercial products may be
* obtained by contacting the authors or the Office of Technology Development
* at the University of North Carolina at Chapel Hill <otd@unc.edu>.
*
* This software program and documentation are copyrighted by the University of
* North Carolina at Chapel Hill. The software program and documentation are
* supplied "as is," without any accompanying services from the University of
* North Carolina at Chapel Hill or the authors. The University of North
* Carolina at Chapel Hill and the authors do not warrant that the operation of
* the program will be uninterrupted or error-free. The end-user understands
* that the program was developed for research purposes and is advised not to
* rely exclusively on the program for any reason.
*
* IN NO EVENT SHALL THE UNIVERSITY OF NORTH CAROLINA AT CHAPEL HILL OR THE
* AUTHORS BE LIABLE TO ANY PARTY FOR DIRECT, INDIRECT, SPECIAL, INCIDENTAL,
* OR CONSEQUENTIAL DAMAGES, INCLUDING LOST PROFITS, ARISING OUT OF THE USE OF
* THIS SOFTWARE AND ITS DOCUMENTATION, EVEN IF THE UNIVERSITY OF NORTH CAROLINA
* AT CHAPEL HILL OR THE AUTHORS HAVE BEEN ADVISED OF THE POSSIBILITY OF SUCH
* DAMAGE.
*
* THE UNIVERSITY OF NORTH CAROLINA AT CHAPEL HILL AND THE AUTHORS SPECIFICALLY
* DISCLAIM ANY WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE AND ANY
* STATUTORY WARRANTY OF NON-INFRINGEMENT. THE SOFTWARE PROVIDED HEREUNDER IS ON
* AN "AS IS" BASIS, AND THE UNIVERSITY OF NORTH CAROLINA AT CHAPEL HILL AND
* THE AUTHORS HAVE NO OBLIGATIONS TO PROVIDE MAINTENANCE, SUPPORT, UPDATES,
* ENHANCEMENTS, OR MODIFICATIONS.
*
* Please send all BUG REPORTS to <pavel@cs.unc.edu>.
*
* The authors may be contacted via:
*
* Pavel Krajcevski
* Dept of Computer Science
* 201 S Columbia St
* Frederick P. Brooks, Jr. Computer Science Bldg
* Chapel Hill, NC 27599-3175
* USA
*
* <http://gamma.cs.unc.edu/FasTC/>
*/
#ifndef FASTC_BASE_INCLUDE_IMAGEFWD_H_
#define FASTC_BASE_INCLUDE_IMAGEFWD_H_
#include "TexCompTypes.h"
namespace FasTC {
class Pixel;
template<typename PixelType = Pixel> class Image;
}
#endif // FASTC_BASE_INCLUDE_IMAGEFWD_H_

View file

@ -49,107 +49,156 @@
#include <cassert> #include <cassert>
#include <cmath> #include <cmath>
#include "Pixel.h"
template <typename T> template <typename T>
static inline T sad( const T &a, const T &b ) { static inline T sad( const T &a, const T &b ) {
return (a > b)? a - b : b - a; return (a > b)? a - b : b - a;
} }
Image::Image(const Image &other) namespace FasTC {
template<typename PixelType>
Image<PixelType>::Image(uint32 width, uint32 height)
: m_Width(width)
, m_Height(height)
, m_bBlockStreamOrder(false)
, m_Pixels(new PixelType[GetNumPixels()])
{ }
template<typename PixelType>
Image<PixelType>::Image(uint32 width, uint32 height,
const PixelType *pixels,
bool bBlockStreamOrder)
: m_Width(width)
, m_Height(height)
, m_bBlockStreamOrder(false)
{
if(pixels) {
m_Pixels = new PixelType[GetNumPixels()];
memcpy(m_Pixels, pixels, GetNumPixels() * sizeof(PixelType));
} else {
m_Pixels = 0;
}
}
template<typename PixelType>
Image<PixelType>::Image(const Image<PixelType> &other)
: m_Width(other.m_Width) : m_Width(other.m_Width)
, m_Height(other.m_Height) , m_Height(other.m_Height)
, m_bBlockStreamOrder(other.GetBlockStreamOrder()) , m_bBlockStreamOrder(other.GetBlockStreamOrder())
, m_DataSz(other.m_DataSz) , m_Pixels(new PixelType[GetNumPixels()])
, m_Data(new uint8[m_DataSz])
{ {
if(m_Data) { memcpy(m_Pixels, other.m_Pixels, GetNumPixels() * sizeof(PixelType));
memcpy(m_Data, other.m_Data, m_DataSz);
} else {
fprintf(stderr, "Out of memory!\n");
}
} }
Image::Image(uint32 width, uint32 height, const uint32 *pixels, bool bBlockStreamOrder) template<typename PixelType>
bool Image<PixelType>::ReadPixels(const uint32 *rgba) {
assert(m_Pixels);
for(uint32 i = 0; i < GetNumPixels(); i++) {
m_Pixels[i].Unpack(rgba[i]);
}
return true;
}
template<typename PixelType>
Image<PixelType>::Image(uint32 width, uint32 height, const uint32 *pixels, bool bBlockStreamOrder)
: m_Width(width) : m_Width(width)
, m_Height(height) , m_Height(height)
, m_bBlockStreamOrder(bBlockStreamOrder) , m_bBlockStreamOrder(bBlockStreamOrder)
, m_DataSz(m_Width * m_Height * sizeof(uint32))
{ {
if(pixels) { if(pixels) {
m_Data = new uint8[m_DataSz]; m_Pixels = new PixelType[GetNumPixels()];
memcpy(m_Data, pixels, m_DataSz); ReadPixels(pixels);
} else { } else {
m_Data = NULL; m_Pixels = NULL;
} }
} }
Image::~Image() { template<typename PixelType>
if(m_Data) { Image<PixelType>::~Image() {
delete [] m_Data; if(m_Pixels) {
m_Data = 0; delete [] m_Pixels;
m_Pixels = 0;
} }
} }
Image &Image::operator=(const Image &other) { template<typename PixelType>
Image<PixelType> &Image<PixelType>::operator=(const Image &other) {
m_Width = other.m_Width; m_Width = other.m_Width;
m_Height = other.m_Height; m_Height = other.m_Height;
m_bBlockStreamOrder = other.GetBlockStreamOrder(); m_bBlockStreamOrder = other.GetBlockStreamOrder();
m_DataSz = other.m_DataSz;
if(m_Data) { if(m_Pixels) {
delete [] m_Data; delete [] m_Pixels;
} }
if(other.m_Data) { if(other.m_Pixels) {
m_Data = new uint8[m_DataSz]; m_Pixels = new PixelType[GetNumPixels()];
if(m_Data) if(m_Pixels)
memcpy(m_Data, other.m_Data, m_DataSz); memcpy(m_Pixels, other.m_Pixels, GetNumPixels() * sizeof(PixelType));
else else
fprintf(stderr, "Out of memory!\n"); fprintf(stderr, "Out of memory!\n");
} } else {
else { m_Pixels = NULL;
m_Data = other.m_Data;
} }
return *this; return *this;
} }
double Image::ComputePSNR(Image *other) { template<typename PixelType>
if(!other) PixelType & Image<PixelType>::operator()(uint32 i, uint32 j) {
assert(i < GetWidth());
assert(j < GetHeight());
return m_Pixels[j * GetWidth() + i];
}
template<typename PixelType>
const PixelType & Image<PixelType>::operator()(uint32 i, uint32 j) const {
assert(i < GetWidth());
assert(j < GetHeight());
return m_Pixels[j * GetWidth() + i];
}
template<typename PixelTypeOne, typename PixelTypeTwo>
double ComputePSNR(Image<PixelTypeOne> *img1, Image<PixelTypeTwo> *img2) {
if(!img1 || !img2)
return -1.0; return -1.0;
if(other->GetWidth() != GetWidth() || if(img1->GetWidth() != img2->GetWidth() ||
other->GetHeight() != GetHeight()) { img1->GetHeight() != img2->GetHeight()) {
return -1.0; return -1.0;
} }
// Compute raw 8-bit RGBA data... // Compute raw 8-bit RGBA data...
other->ComputeRGBA(); img1->ComputePixels();
ComputeRGBA(); img2->ComputePixels();
const uint8 *ourData = const PixelTypeOne *ourPixels = img1->GetPixels();
reinterpret_cast<const uint8 *>(GetRGBA()); const PixelTypeTwo *otherPixels = img2->GetPixels();
const uint8 *otherData =
reinterpret_cast<const uint8 *>(other->GetRGBA());
// const double w[3] = { 0.2126, 0.7152, 0.0722 }; // const double w[3] = { 0.2126, 0.7152, 0.0722 };
const double w[3] = { 1.0, 1.0, 1.0 }; const double w[3] = { 1.0, 1.0, 1.0 };
double mse = 0.0; double mse = 0.0;
const uint32 imageSz = GetWidth() * GetHeight() * 4; const uint32 imageSz = img1->GetNumPixels();
for(uint32 i = 0; i < imageSz; i+=4) { for(uint32 i = 0; i < imageSz; i++) {
const unsigned char *pixelDataRaw = ourData + i; uint32 ourPixel = ourPixels[i].Pack();
const unsigned char *pixelDataUncomp = otherData + i; uint32 otherPixel = otherPixels[i].Pack();
double r[4], u[4]; double r[4], u[4];
for(uint32 c = 0; c < 4; c++) { for(uint32 c = 0; c < 4; c++) {
uint32 shift = c * 8;
if(c == 3) { if(c == 3) {
r[c] = pixelDataRaw[c] / 255.0; r[c] = static_cast<double>((ourPixel >> shift) & 0xFF) / 255.0;
u[c] = pixelDataUncomp[c] / 255.0; u[c] = static_cast<double>((otherPixel >> shift) & 0xFF) / 255.0;
} else { } else {
r[c] = static_cast<double>(pixelDataRaw[c]) * w[c]; r[c] = static_cast<double>((ourPixel >> shift) & 0xFF) * w[c];
u[c] = static_cast<double>(pixelDataUncomp[c]) * w[c]; u[c] = static_cast<double>((otherPixel >> shift) & 0xFF) * w[c];
} }
} }
@ -159,19 +208,22 @@ double Image::ComputePSNR(Image *other) {
} }
} }
mse /= GetWidth() * GetHeight(); mse /= img1->GetWidth() * img1->GetHeight();
const double C = 255.0 * 255.0; const double C = 255.0 * 255.0;
double maxi = (w[0]*w[0] + w[1]*w[1] + w[2]*w[2]) * C; double maxi = (w[0]*w[0] + w[1]*w[1] + w[2]*w[2]) * C;
return 10 * log10(maxi/mse); return 10 * log10(maxi/mse);
} }
template double ComputePSNR(Image<Pixel> *, Image<Pixel> *);
// !FIXME! These won't work for non-RGBA8 data. // !FIXME! These won't work for non-RGBA8 data.
void Image::ConvertToBlockStreamOrder() { template<typename PixelType>
if(m_bBlockStreamOrder || !m_Data) void Image<PixelType>::ConvertToBlockStreamOrder() {
if(m_bBlockStreamOrder || !m_Pixels)
return; return;
uint32 *newPixelData = new uint32[GetWidth() * GetHeight() * 4]; PixelType *newPixelData = new PixelType[GetWidth() * GetHeight()];
for(uint32 j = 0; j < GetHeight(); j+=4) { for(uint32 j = 0; j < GetHeight(); j+=4) {
for(uint32 i = 0; i < GetWidth(); i+=4) { for(uint32 i = 0; i < GetWidth(); i+=4) {
uint32 blockX = i / 4; uint32 blockX = i / 4;
@ -182,22 +234,22 @@ void Image::ConvertToBlockStreamOrder() {
for(uint32 t = 0; t < 16; t++) { for(uint32 t = 0; t < 16; t++) {
uint32 x = i + t % 4; uint32 x = i + t % 4;
uint32 y = j + t / 4; uint32 y = j + t / 4;
newPixelData[offset + t] = newPixelData[offset + t] = m_Pixels[y*GetWidth() + x];
reinterpret_cast<uint32 *>(m_Data)[y*GetWidth() + x];
} }
} }
} }
delete m_Data; delete m_Pixels;
m_Data = reinterpret_cast<uint8 *>(newPixelData); m_Pixels = newPixelData;
m_bBlockStreamOrder = true; m_bBlockStreamOrder = true;
} }
void Image::ConvertFromBlockStreamOrder() { template<typename PixelType>
if(!m_bBlockStreamOrder || !m_Data) void Image<PixelType>::ConvertFromBlockStreamOrder() {
if(!m_bBlockStreamOrder || !m_Pixels)
return; return;
uint32 *newPixelData = new uint32[GetWidth() * GetHeight() * 4]; PixelType *newPixelData = new PixelType[GetWidth() * GetHeight()];
for(uint32 j = 0; j < GetHeight(); j+=4) { for(uint32 j = 0; j < GetHeight(); j+=4) {
for(uint32 i = 0; i < GetWidth(); i+=4) { for(uint32 i = 0; i < GetWidth(); i+=4) {
uint32 blockX = i / 4; uint32 blockX = i / 4;
@ -208,13 +260,33 @@ void Image::ConvertFromBlockStreamOrder() {
for(uint32 t = 0; t < 16; t++) { for(uint32 t = 0; t < 16; t++) {
uint32 x = i + t % 4; uint32 x = i + t % 4;
uint32 y = j + t / 4; uint32 y = j + t / 4;
newPixelData[y*GetWidth() + x] = newPixelData[y*GetWidth() + x] = m_Pixels[offset + t];
reinterpret_cast<uint32 *>(m_Data)[offset + t];
} }
} }
} }
delete m_Data; delete m_Pixels;
m_Data = reinterpret_cast<uint8 *>(newPixelData); m_Pixels = newPixelData;
m_bBlockStreamOrder = false; m_bBlockStreamOrder = false;
} }
template<typename PixelType>
void Image<PixelType>::SetImageData(uint32 width, uint32 height, PixelType *data) {
if(m_Pixels) {
delete m_Pixels;
}
if(!data) {
width = 0;
height = 0;
m_Pixels = NULL;
} else {
m_Width = width;
m_Height = height;
m_Pixels = data;
}
}
template class Image<Pixel>;
} // namespace FasTC

View file

@ -53,11 +53,11 @@ INCLUDE_DIRECTORIES(${FasTC_SOURCE_DIR}/Base/include)
INCLUDE_DIRECTORIES(${FasTC_SOURCE_DIR}/GTest/include) INCLUDE_DIRECTORIES(${FasTC_SOURCE_DIR}/GTest/include)
SET(TESTS SET(TESTS
Pixel Pixel Image
) )
FOREACH(TEST ${TESTS}) FOREACH(TEST ${TESTS})
SET(TEST_NAME Test${TEST}) SET(TEST_NAME Test_Base_${TEST})
SET(TEST_MODULE Test${TEST}.cpp) SET(TEST_MODULE Test${TEST}.cpp)
ADD_EXECUTABLE(${TEST_NAME} ${TEST_MODULE}) ADD_EXECUTABLE(${TEST_NAME} ${TEST_MODULE})
TARGET_LINK_LIBRARIES(${TEST_NAME} FasTCBase) TARGET_LINK_LIBRARIES(${TEST_NAME} FasTCBase)

121
Base/test/TestImage.cpp Normal file
View file

@ -0,0 +1,121 @@
/* FasTC
* Copyright (c) 2013 University of North Carolina at Chapel Hill.
* All rights reserved.
*
* Permission to use, copy, modify, and distribute this software and its
* documentation for educational, research, and non-profit purposes, without
* fee, and without a written agreement is hereby granted, provided that the
* above copyright notice, this paragraph, and the following four paragraphs
* appear in all copies.
*
* Permission to incorporate this software into commercial products may be
* obtained by contacting the authors or the Office of Technology Development
* at the University of North Carolina at Chapel Hill <otd@unc.edu>.
*
* This software program and documentation are copyrighted by the University of
* North Carolina at Chapel Hill. The software program and documentation are
* supplied "as is," without any accompanying services from the University of
* North Carolina at Chapel Hill or the authors. The University of North
* Carolina at Chapel Hill and the authors do not warrant that the operation of
* the program will be uninterrupted or error-free. The end-user understands
* that the program was developed for research purposes and is advised not to
* rely exclusively on the program for any reason.
*
* IN NO EVENT SHALL THE UNIVERSITY OF NORTH CAROLINA AT CHAPEL HILL OR THE
* AUTHORS BE LIABLE TO ANY PARTY FOR DIRECT, INDIRECT, SPECIAL, INCIDENTAL,
* OR CONSEQUENTIAL DAMAGES, INCLUDING LOST PROFITS, ARISING OUT OF THE USE OF
* THIS SOFTWARE AND ITS DOCUMENTATION, EVEN IF THE UNIVERSITY OF NORTH CAROLINA
* AT CHAPEL HILL OR THE AUTHORS HAVE BEEN ADVISED OF THE POSSIBILITY OF SUCH
* DAMAGE.
*
* THE UNIVERSITY OF NORTH CAROLINA AT CHAPEL HILL AND THE AUTHORS SPECIFICALLY
* DISCLAIM ANY WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE AND ANY
* STATUTORY WARRANTY OF NON-INFRINGEMENT. THE SOFTWARE PROVIDED HEREUNDER IS ON
* AN "AS IS" BASIS, AND THE UNIVERSITY OF NORTH CAROLINA AT CHAPEL HILL AND
* THE AUTHORS HAVE NO OBLIGATIONS TO PROVIDE MAINTENANCE, SUPPORT, UPDATES,
* ENHANCEMENTS, OR MODIFICATIONS.
*
* Please send all BUG REPORTS to <pavel@cs.unc.edu>.
*
* The authors may be contacted via:
*
* Pavel Krajcevski
* Dept of Computer Science
* 201 S Columbia St
* Frederick P. Brooks, Jr. Computer Science Bldg
* Chapel Hill, NC 27599-3175
* USA
*
* <http://gamma.cs.unc.edu/FasTC/>
*/
#include "gtest/gtest.h"
#include "Image.h"
#include "Pixel.h"
#include "Utils.h"
#include <cstdlib>
TEST(Image, NonSpecificConstructor) {
FasTC::Pixel p;
FasTC::Image<FasTC::Pixel> img (4, 4);
for(uint32 i = 0; i < 4; i++) {
for(uint32 j = 0; j < 4; j++) {
EXPECT_TRUE(img(i, j) == p);
}
}
}
TEST(Image, SpecificConstructor) {
FasTC::Pixel pxs[16];
for(uint32 i = 0; i < 4; i++) {
for(uint32 j = 0; j < 4; j++) {
pxs[j*4 + i].R() = i;
pxs[j*4 + i].G() = j;
}
}
FasTC::Image<FasTC::Pixel> img(4, 4, pxs);
for(uint32 i = 0; i < 4; i++) {
for(uint32 j = 0; j < 4; j++) {
EXPECT_TRUE(img(i, j) == pxs[j*4 + i]);
}
}
}
TEST(Image, CopyConstructor) {
FasTC::Pixel pxs[16];
for(uint32 i = 0; i < 4; i++) {
for(uint32 j = 0; j < 4; j++) {
pxs[j*4 + i].R() = i;
pxs[j*4 + i].G() = j;
}
}
FasTC::Image<FasTC::Pixel> img(4, 4, pxs);
FasTC::Image<FasTC::Pixel> img2(img);
for(uint32 i = 0; i < 4; i++) {
for(uint32 j = 0; j < 4; j++) {
EXPECT_TRUE(img2(i, j) == pxs[j*4 + i]);
}
}
}
TEST(Image, AssignmentOperator) {
FasTC::Pixel pxs[16];
for(uint32 i = 0; i < 4; i++) {
for(uint32 j = 0; j < 4; j++) {
pxs[j*4 + i].R() = i;
pxs[j*4 + i].G() = j;
}
}
FasTC::Image<FasTC::Pixel> img(4, 4, pxs);
FasTC::Image<FasTC::Pixel> img2 = img;
for(uint32 i = 0; i < 4; i++) {
for(uint32 j = 0; j < 4; j++) {
EXPECT_TRUE(img2(i, j) == pxs[j*4 + i]);
}
}
}

82
Base/test/Utils.h Normal file
View file

@ -0,0 +1,82 @@
/* FasTC
* Copyright (c) 2013 University of North Carolina at Chapel Hill.
* All rights reserved.
*
* Permission to use, copy, modify, and distribute this software and its
* documentation for educational, research, and non-profit purposes, without
* fee, and without a written agreement is hereby granted, provided that the
* above copyright notice, this paragraph, and the following four paragraphs
* appear in all copies.
*
* Permission to incorporate this software into commercial products may be
* obtained by contacting the authors or the Office of Technology Development
* at the University of North Carolina at Chapel Hill <otd@unc.edu>.
*
* This software program and documentation are copyrighted by the University of
* North Carolina at Chapel Hill. The software program and documentation are
* supplied "as is," without any accompanying services from the University of
* North Carolina at Chapel Hill or the authors. The University of North
* Carolina at Chapel Hill and the authors do not warrant that the operation of
* the program will be uninterrupted or error-free. The end-user understands
* that the program was developed for research purposes and is advised not to
* rely exclusively on the program for any reason.
*
* IN NO EVENT SHALL THE UNIVERSITY OF NORTH CAROLINA AT CHAPEL HILL OR THE
* AUTHORS BE LIABLE TO ANY PARTY FOR DIRECT, INDIRECT, SPECIAL, INCIDENTAL,
* OR CONSEQUENTIAL DAMAGES, INCLUDING LOST PROFITS, ARISING OUT OF THE USE OF
* THIS SOFTWARE AND ITS DOCUMENTATION, EVEN IF THE UNIVERSITY OF NORTH CAROLINA
* AT CHAPEL HILL OR THE AUTHORS HAVE BEEN ADVISED OF THE POSSIBILITY OF SUCH
* DAMAGE.
*
* THE UNIVERSITY OF NORTH CAROLINA AT CHAPEL HILL AND THE AUTHORS SPECIFICALLY
* DISCLAIM ANY WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE AND ANY
* STATUTORY WARRANTY OF NON-INFRINGEMENT. THE SOFTWARE PROVIDED HEREUNDER IS ON
* AN "AS IS" BASIS, AND THE UNIVERSITY OF NORTH CAROLINA AT CHAPEL HILL AND
* THE AUTHORS HAVE NO OBLIGATIONS TO PROVIDE MAINTENANCE, SUPPORT, UPDATES,
* ENHANCEMENTS, OR MODIFICATIONS.
*
* Please send all BUG REPORTS to <pavel@cs.unc.edu>.
*
* The authors may be contacted via:
*
* Pavel Krajcevski
* Dept of Computer Science
* 201 S Columbia St
* Frederick P. Brooks, Jr. Computer Science Bldg
* Chapel Hill, NC 27599-3175
* USA
*
* <http://gamma.cs.unc.edu/FasTC/>
*/
#ifndef PVRTCENCODER_TEST_TESTUTILS_H_
#define PVRTCENCODER_TEST_TESTUTILS_H_
#include "TexCompTypes.h"
class PixelPrinter {
private:
uint32 m_PixelValue;
public:
explicit PixelPrinter(uint32 p) : m_PixelValue(p) { }
bool operator==(const PixelPrinter &other) const {
return other.m_PixelValue == this->m_PixelValue;
}
uint32 Value() const { return m_PixelValue; }
};
inline ::std::ostream& operator<<(::std::ostream& os, const PixelPrinter& pp) {
uint32 p = pp.Value();
uint32 r = p & 0xFF;
uint32 g = (p >> 8) & 0xFF;
uint32 b = (p >> 16) & 0xFF;
uint32 a = (p >> 24) & 0xFF;
return os <<
"R: 0x" << ::std::hex << r << " " <<
"G: 0x" << ::std::hex << g << " " <<
"B: 0x" << ::std::hex << b << " " <<
"A: 0x" << ::std::hex << a;
}
#endif // PVRTCENCODER_TEST_TESTUTILS_H_

View file

@ -208,7 +208,7 @@ int main(int argc, char **argv) {
return 1; return 1;
} }
Image img = Image(*file.GetImage()); FasTC::Image<> img(*file.GetImage());
if(format == eCompressionFormat_PVRTC) { if(format == eCompressionFormat_PVRTC) {
img.SetBlockStreamOrder(false); img.SetBlockStreamOrder(false);
} }

View file

@ -57,10 +57,10 @@ enum ECompressionFormat {
#include "Image.h" #include "Image.h"
class CompressedImage : public Image { class CompressedImage : public FasTC::Image<FasTC::Pixel> {
private: private:
ECompressionFormat m_Format; ECompressionFormat m_Format;
uint32 *m_RGBAData; uint8 *m_CompressedData;
public: public:
CompressedImage(const CompressedImage &); CompressedImage(const CompressedImage &);
@ -78,12 +78,11 @@ class CompressedImage : public Image {
virtual ~CompressedImage(); virtual ~CompressedImage();
virtual Image *Clone() const { virtual FasTC::Image<FasTC::Pixel> *Clone() const {
return new CompressedImage(*this); return new CompressedImage(*this);
} }
virtual void ComputeRGBA(); virtual void ComputePixels();
virtual const uint32 *GetRGBA() const { return m_RGBAData; }
static uint32 GetCompressedSize(uint32 uncompressedSize, ECompressionFormat format); static uint32 GetCompressedSize(uint32 uncompressedSize, ECompressionFormat format);
static uint32 GetUncompressedSize(uint32 compressedSize, ECompressionFormat format) { static uint32 GetUncompressedSize(uint32 compressedSize, ECompressionFormat format) {
@ -91,6 +90,13 @@ class CompressedImage : public Image {
return compressedSize * (compressedSize / cmp); return compressedSize * (compressedSize / cmp);
} }
uint32 GetCompressedSize() const {
return GetCompressedSize(GetUncompressedSize(), m_Format);
}
uint32 GetUncompressedSize() const {
return GetWidth() * GetHeight() * sizeof(uint32);
}
// Decompress the compressed image data into outBuf. outBufSz is expected // Decompress the compressed image data into outBuf. outBufSz is expected
// to be the proper size determined by the width, height, and format. // to be the proper size determined by the width, height, and format.
// !FIXME! We should have a function to explicitly return the in/out buf // !FIXME! We should have a function to explicitly return the in/out buf

View file

@ -48,9 +48,9 @@
#include "CompressionJob.h" #include "CompressionJob.h"
#include <iosfwd> #include <iosfwd>
#include "ImageFwd.h"
// Forward declarations // Forward declarations
class Image;
class ImageFile; class ImageFile;
struct SCompressionSettings { struct SCompressionSettings {
@ -91,7 +91,8 @@ struct SCompressionSettings {
std::ostream *logStream; std::ostream *logStream;
}; };
extern CompressedImage *CompressImage(Image *img, const SCompressionSettings &settings); template<typename PixelType>
extern CompressedImage *CompressImage(FasTC::Image<PixelType> *img, const SCompressionSettings &settings);
extern bool CompressImageData( extern bool CompressImageData(
const unsigned char *data, const unsigned char *data,

View file

@ -48,6 +48,8 @@
#include <stdio.h> #include <stdio.h>
#include <assert.h> #include <assert.h>
#include "Pixel.h"
#include "TexCompTypes.h" #include "TexCompTypes.h"
#include "BC7Compressor.h" #include "BC7Compressor.h"
#include "PVRTCCompressor.h" #include "PVRTCCompressor.h"
@ -55,11 +57,12 @@
CompressedImage::CompressedImage( const CompressedImage &other ) CompressedImage::CompressedImage( const CompressedImage &other )
: Image(other) : Image(other)
, m_Format(other.m_Format) , m_Format(other.m_Format)
, m_RGBAData(0) , m_CompressedData(0)
{ {
if(other.m_RGBAData) { if(other.m_CompressedData) {
m_RGBAData = new uint32[GetWidth() * GetHeight()]; uint32 compressedSz = GetCompressedSize();
memcpy(m_RGBAData, other.m_RGBAData, sizeof(uint32) * GetWidth() * GetHeight()); m_CompressedData = new uint8[compressedSz];
memcpy(m_CompressedData, other.m_CompressedData, compressedSz);
} }
} }
@ -69,47 +72,44 @@ CompressedImage::CompressedImage(
const ECompressionFormat format, const ECompressionFormat format,
const unsigned char *data const unsigned char *data
) )
: Image(width, height, NULL, format != eCompressionFormat_PVRTC) : FasTC::Image<>(width, height,
reinterpret_cast<uint32 *>(NULL),
format != eCompressionFormat_PVRTC)
, m_Format(format) , m_Format(format)
, m_RGBAData(0) , m_CompressedData(0)
{ {
m_DataSz = GetCompressedSize(GetWidth() * GetHeight() * 4, m_Format); uint32 cmpSz = GetCompressedSize();
if(m_DataSz > 0) { if(cmpSz > 0) {
assert(!m_Data); assert(!m_CompressedData);
m_Data = new unsigned char[m_DataSz]; m_CompressedData = new uint8[cmpSz];
memcpy(m_Data, data, m_DataSz); memcpy(m_CompressedData, data, cmpSz);
} }
} }
CompressedImage &CompressedImage::operator=(const CompressedImage &other) { CompressedImage &CompressedImage::operator=(const CompressedImage &other) {
Image::operator=(other); Image::operator=(other);
m_Format = other.m_Format; m_Format = other.m_Format;
if(other.m_RGBAData) { if(other.m_CompressedData) {
m_RGBAData = new uint32[GetWidth() * GetHeight()]; uint32 cmpSz = GetCompressedSize();
memcpy(m_RGBAData, other.m_RGBAData, sizeof(uint32) * GetWidth() * GetHeight()); m_CompressedData = new uint8[cmpSz];
memcpy(m_CompressedData, other.m_CompressedData, cmpSz);
} }
return *this; return *this;
} }
CompressedImage::~CompressedImage() { CompressedImage::~CompressedImage() {
if(m_RGBAData) { if(m_CompressedData) {
delete m_RGBAData; delete m_CompressedData;
m_RGBAData = NULL; m_CompressedData = NULL;
} }
} }
bool CompressedImage::DecompressImage(unsigned char *outBuf, unsigned int outBufSz) const { bool CompressedImage::DecompressImage(unsigned char *outBuf, unsigned int outBufSz) const {
// First make sure that we have enough data assert(outBufSz == GetUncompressedSize());
uint32 dataSz = GetUncompressedSize(m_DataSz, m_Format);
if(dataSz > outBufSz) {
fprintf(stderr, "Not enough space to store entire decompressed image! "
"Got %d bytes, but need %d!\n", outBufSz, dataSz);
assert(false);
return false;
}
DecompressionJob dj (m_Data, outBuf, GetWidth(), GetHeight()); uint8 *byteData = reinterpret_cast<uint8 *>(m_CompressedData);
DecompressionJob dj (byteData, outBuf, GetWidth(), GetHeight());
switch(m_Format) { switch(m_Format) {
case eCompressionFormat_PVRTC: case eCompressionFormat_PVRTC:
{ {
@ -135,15 +135,20 @@ bool CompressedImage::DecompressImage(unsigned char *outBuf, unsigned int outBuf
return true; return true;
} }
void CompressedImage::ComputeRGBA() { void CompressedImage::ComputePixels() {
if(m_RGBAData) { uint32 unCompSz = GetWidth() * GetHeight() * 4;
delete m_RGBAData; uint8 *unCompBuf = new uint8[unCompSz];
DecompressImage(unCompBuf, unCompSz);
uint32 * newPixelBuf = reinterpret_cast<uint32 *>(unCompBuf);
FasTC::Pixel *newPixels = new FasTC::Pixel[GetWidth() * GetHeight()];
for(uint32 i = 0; i < GetWidth() * GetHeight(); i++) {
newPixels[i].Unpack(newPixelBuf[i]);
} }
m_RGBAData = new uint32[GetWidth() * GetHeight()];
uint8 *pixelData = reinterpret_cast<uint8 *>(m_RGBAData); SetImageData(GetWidth(), GetHeight(), newPixels);
DecompressImage(pixelData, GetWidth() * GetHeight() * 4);
} }
uint32 CompressedImage::GetCompressedSize(uint32 uncompressedSize, ECompressionFormat format) { uint32 CompressedImage::GetCompressedSize(uint32 uncompressedSize, ECompressionFormat format) {
@ -167,4 +172,3 @@ uint32 CompressedImage::GetCompressedSize(uint32 uncompressedSize, ECompressionF
return cmpDataSzNeeded; return cmpDataSzNeeded;
} }

View file

@ -54,6 +54,7 @@
#include "CompressionFuncs.h" #include "CompressionFuncs.h"
#include "Image.h" #include "Image.h"
#include "ImageFile.h" #include "ImageFile.h"
#include "Pixel.h"
#include "PVRTCCompressor.h" #include "PVRTCCompressor.h"
#include "Thread.h" #include "Thread.h"
#include "ThreadGroup.h" #include "ThreadGroup.h"
@ -349,8 +350,9 @@ static double CompressImageWithWorkerQueue(
return cmpTimeTotal / double(settings.iNumCompressions); return cmpTimeTotal / double(settings.iNumCompressions);
} }
template<typename PixelType>
CompressedImage *CompressImage( CompressedImage *CompressImage(
Image *img, const SCompressionSettings &settings FasTC::Image<PixelType> *img, const SCompressionSettings &settings
) { ) {
if(!img) return NULL; if(!img) return NULL;
@ -359,6 +361,7 @@ CompressedImage *CompressImage(
CompressedImage *outImg = NULL; CompressedImage *outImg = NULL;
const unsigned int dataSz = w * h * 4; const unsigned int dataSz = w * h * 4;
uint32 *data = new uint32[dataSz / 4];
assert(dataSz > 0); assert(dataSz > 0);
@ -366,24 +369,33 @@ CompressedImage *CompressImage(
uint32 cmpDataSz = CompressedImage::GetCompressedSize(dataSz, settings.format); uint32 cmpDataSz = CompressedImage::GetCompressedSize(dataSz, settings.format);
// Make sure that we have RGBA data... // Make sure that we have RGBA data...
img->ComputeRGBA(); img->ComputePixels();
const PixelType *pixels = img->GetPixels();
for(uint32 i = 0; i < img->GetNumPixels(); i++) {
data[i] = pixels[i].Pack();
}
unsigned char *cmpData = new unsigned char[cmpDataSz]; unsigned char *cmpData = new unsigned char[cmpDataSz];
const uint8 *pixelData = reinterpret_cast<const uint8 *>(img->GetRGBA()); CompressImageData(reinterpret_cast<uint8 *>(data), w, h, cmpData, cmpDataSz, settings);
CompressImageData(pixelData, w, h, cmpData, cmpDataSz, settings);
outImg = new CompressedImage(w, h, settings.format, cmpData); outImg = new CompressedImage(w, h, settings.format, cmpData);
delete [] data;
delete [] cmpData; delete [] cmpData;
return outImg; return outImg;
} }
// !FIXME! Ideally, we wouldn't have to do this because there would be a way to instantiate this
// function in the header or using some fancy template metaprogramming. I can't think of the way
// at the moment.
template CompressedImage *CompressImage(FasTC::Image<FasTC::Pixel> *, const SCompressionSettings &settings);
bool CompressImageData( bool CompressImageData(
const unsigned char *data, const uint8 *data,
const unsigned int width, const uint32 width,
const unsigned int height, const uint32 height,
unsigned char *cmpData, uint8 *compressedData,
const unsigned int cmpDataSz, const uint32 cmpDataSz,
const SCompressionSettings &settings const SCompressionSettings &settings
) { ) {
@ -417,14 +429,14 @@ bool CompressImageData(
} }
// Allocate data based on the compression method // Allocate data based on the compression method
uint32 cmpDataSzNeeded = uint32 compressedDataSzNeeded =
CompressedImage::GetCompressedSize(dataSz, settings.format); CompressedImage::GetCompressedSize(dataSz, settings.format);
if(cmpDataSzNeeded == 0) { if(compressedDataSzNeeded == 0) {
ReportError("Unknown compression format"); ReportError("Unknown compression format");
return false; return false;
} }
else if(cmpDataSzNeeded > cmpDataSz) { else if(compressedDataSzNeeded > cmpDataSz) {
ReportError("Not enough space for compressed data!"); ReportError("Not enough space for compressed data!");
return false; return false;
} }
@ -436,15 +448,15 @@ bool CompressImageData(
if(numThreads > 1) { if(numThreads > 1) {
if(settings.bUseAtomics) { if(settings.bUseAtomics) {
cmpMSTime = CompressImageWithAtomics(data, width, height, settings, cmpData); cmpMSTime = CompressImageWithAtomics(data, width, height, settings, compressedData);
} else if(settings.iJobSize > 0) { } else if(settings.iJobSize > 0) {
cmpMSTime = CompressImageWithWorkerQueue(data, dataSz, settings, cmpData); cmpMSTime = CompressImageWithWorkerQueue(data, dataSz, settings, compressedData);
} else { } else {
cmpMSTime = CompressImageWithThreads(data, dataSz, settings, cmpData); cmpMSTime = CompressImageWithThreads(data, dataSz, settings, compressedData);
} }
} }
else { else {
cmpMSTime = CompressImageInSerial(data, width, height, settings, cmpData); cmpMSTime = CompressImageInSerial(data, width, height, settings, compressedData);
} }
// Report compression time // Report compression time

View file

@ -47,19 +47,23 @@
#include "ImageFileFormat.h" #include "ImageFileFormat.h"
#include "TexCompTypes.h" #include "TexCompTypes.h"
namespace FasTC {
class Pixel;
}
class ImageWriter { class ImageWriter {
protected: protected:
const uint8 *m_PixelData; const FasTC::Pixel *m_Pixels;
uint32 m_RawFileDataSz; uint32 m_RawFileDataSz;
uint8 *m_RawFileData; uint8 *m_RawFileData;
uint32 m_Width; uint32 m_Width;
uint32 m_Height; uint32 m_Height;
ImageWriter(const int width, const int height, const uint8 *rawData) ImageWriter(const int width, const int height, const FasTC::Pixel *rawData)
: m_PixelData(rawData) : m_Pixels(rawData)
, m_RawFileDataSz(256) , m_RawFileDataSz(256)
, m_RawFileData(new uint8[m_RawFileDataSz]) , m_RawFileData(new uint8[m_RawFileDataSz])
, m_Width(width), m_Height(height) , m_Width(width), m_Height(height)
@ -77,7 +81,7 @@ class ImageWriter {
uint32 GetWidth() const { return m_Width; } uint32 GetWidth() const { return m_Width; }
uint32 GetHeight() const { return m_Height; } uint32 GetHeight() const { return m_Height; }
uint32 GetImageDataSz() const { return m_Width * m_Height * 4; } uint32 GetImageDataSz() const { return m_Width * m_Height * sizeof(uint32); }
uint32 GetRawFileDataSz() const { return m_RawFileDataSz; } uint32 GetRawFileDataSz() const { return m_RawFileDataSz; }
uint8 *GetRawFileData() const { return m_RawFileData; } uint8 *GetRawFileData() const { return m_RawFileData; }
virtual bool WriteImage() = 0; virtual bool WriteImage() = 0;

View file

@ -46,9 +46,9 @@
#include "TexCompTypes.h" #include "TexCompTypes.h"
#include "ImageFileFormat.h" #include "ImageFileFormat.h"
#include "ImageFwd.h"
// Forward declare // Forward declare
class Image;
class CompressedImage; class CompressedImage;
struct SCompressionSettings; struct SCompressionSettings;
@ -66,13 +66,13 @@ public:
// Creates an imagefile with the corresponding image data. This is ready // Creates an imagefile with the corresponding image data. This is ready
// to be written to disk with the passed filename. // to be written to disk with the passed filename.
ImageFile(const char *filename, EImageFileFormat format, const Image &); ImageFile(const char *filename, EImageFileFormat format, const FasTC::Image<> &);
~ImageFile(); ~ImageFile();
unsigned int GetWidth() const { return m_Width; } unsigned int GetWidth() const { return m_Width; }
unsigned int GetHeight() const { return m_Height; } unsigned int GetHeight() const { return m_Height; }
Image *GetImage() const { return m_Image; } FasTC::Image<> *GetImage() const { return m_Image; }
// Loads the image into memory. If this function returns true, then a valid // Loads the image into memory. If this function returns true, then a valid
// m_Image will be created and available. // m_Image will be created and available.
@ -91,12 +91,12 @@ public:
const EImageFileFormat m_FileFormat; const EImageFileFormat m_FileFormat;
Image *m_Image; FasTC::Image<> *m_Image;
static unsigned char *ReadFileData(const CHAR *filename); static unsigned char *ReadFileData(const CHAR *filename);
static bool WriteImageDataToFile(const uint8 *data, const uint32 dataSz, const CHAR *filename); static bool WriteImageDataToFile(const uint8 *data, const uint32 dataSz, const CHAR *filename);
static EImageFileFormat DetectFileFormat(const CHAR *filename); static EImageFileFormat DetectFileFormat(const CHAR *filename);
Image *LoadImage(const unsigned char *rawImageData) const; FasTC::Image<> *LoadImage(const unsigned char *rawImageData) const;
}; };
#endif // _IMAGE_FILE_H_ #endif // _IMAGE_FILE_H_

View file

@ -105,7 +105,7 @@ ImageFile::ImageFile(const CHAR *filename, EImageFileFormat format)
strncpy(m_Filename, filename, kMaxFilenameSz); strncpy(m_Filename, filename, kMaxFilenameSz);
} }
ImageFile::ImageFile(const char *filename, EImageFileFormat format, const Image &image) ImageFile::ImageFile(const char *filename, EImageFileFormat format, const FasTC::Image<> &image)
: m_FileFormat(format) : m_FileFormat(format)
, m_Image(image.Clone()) , m_Image(image.Clone())
{ {
@ -165,7 +165,7 @@ bool ImageFile::Write() {
return true; return true;
} }
Image *ImageFile::LoadImage(const unsigned char *rawImageData) const { FasTC::Image<> *ImageFile::LoadImage(const unsigned char *rawImageData) const {
ImageLoader *loader = NULL; ImageLoader *loader = NULL;
switch(m_FileFormat) { switch(m_FileFormat) {
@ -207,7 +207,7 @@ Image *ImageFile::LoadImage(const unsigned char *rawImageData) const {
} }
uint32 *pixels = reinterpret_cast<uint32 *>(pixelData); uint32 *pixels = reinterpret_cast<uint32 *>(pixelData);
Image *i = new Image(loader->GetWidth(), loader->GetHeight(), pixels, true); FasTC::Image<> *i = new FasTC::Image<>(loader->GetWidth(), loader->GetHeight(), pixels, true);
// Cleanup // Cleanup
delete loader; delete loader;

View file

@ -42,6 +42,7 @@
*/ */
#include "ImageWriter.h" #include "ImageWriter.h"
#include "Pixel.h"
uint32 ImageWriter::GetChannelForPixel(uint32 x, uint32 y, uint32 ch) { uint32 ImageWriter::GetChannelForPixel(uint32 x, uint32 y, uint32 ch) {
@ -58,10 +59,8 @@ uint32 ImageWriter::GetChannelForPixel(uint32 x, uint32 y, uint32 ch) {
const uint32 blockOffsetY = y % 4; const uint32 blockOffsetY = y % 4;
const uint32 pixelOffset = blockOffsetY * 4 + blockOffsetX; const uint32 pixelOffset = blockOffsetY * 4 + blockOffsetX;
// There are 16 pixels per block and bytes per pixel... // There are 16 pixels per block...
uint32 dataOffset = blockIdx * 4 * 16; uint32 dataOffset = blockIdx * 16 + pixelOffset;
dataOffset += 4 * pixelOffset;
dataOffset += ch;
return m_PixelData[dataOffset]; return m_Pixels[dataOffset].Component(ch);
} }

View file

@ -43,14 +43,15 @@
#include "ImageWriterPNG.h" #include "ImageWriterPNG.h"
#include <stdlib.h> #include <cstdlib>
#include <stdio.h> #include <cstdio>
#include <string.h> #include <cstring>
#include "Image.h"
#include <png.h> #include <png.h>
#include "Image.h"
#include "Pixel.h"
class PNGStreamWriter { class PNGStreamWriter {
public: public:
static void WriteDataToStream( static void WriteDataToStream(
@ -84,13 +85,13 @@ public:
}; };
ImageWriterPNG::ImageWriterPNG(Image &im) ImageWriterPNG::ImageWriterPNG(FasTC::Image<> &im)
: ImageWriter(im.GetWidth(), im.GetHeight(), im.RawData()) : ImageWriter(im.GetWidth(), im.GetHeight(), im.GetPixels())
, m_bBlockStreamOrder(im.GetBlockStreamOrder()) , m_bBlockStreamOrder(im.GetBlockStreamOrder())
, m_StreamPosition(0) , m_StreamPosition(0)
{ {
im.ComputeRGBA(); im.ComputePixels();
m_PixelData = reinterpret_cast<const uint8 *>(im.GetRGBA()); m_Pixels = im.GetPixels();
} }
bool ImageWriterPNG::WriteImage() { bool ImageWriterPNG::WriteImage() {
@ -136,8 +137,7 @@ bool ImageWriterPNG::WriteImage() {
*row++ = GetChannelForPixel(x, y, ch); *row++ = GetChannelForPixel(x, y, ch);
} }
} else { } else {
reinterpret_cast<uint32 *>(row)[x] = reinterpret_cast<uint32 *>(row)[x] = m_Pixels[y * m_Width + x].Pack();
reinterpret_cast<const uint32 *>(m_PixelData)[y * m_Width + x];
} }
} }
} }

View file

@ -45,12 +45,12 @@
#define _IMAGE_WRITER_PNG_H_ #define _IMAGE_WRITER_PNG_H_
#include "ImageWriter.h" #include "ImageWriter.h"
#include "ImageFwd.h"
// Forward Declare // Forward Declare
class Image;
class ImageWriterPNG : public ImageWriter { class ImageWriterPNG : public ImageWriter {
public: public:
ImageWriterPNG(Image &); ImageWriterPNG(FasTC::Image<> &);
virtual ~ImageWriterPNG() { } virtual ~ImageWriterPNG() { }
virtual bool WriteImage(); virtual bool WriteImage();

View file

@ -57,14 +57,14 @@ INCLUDE_DIRECTORIES(${FasTC_SOURCE_DIR}/Base/include)
SET( HEADERS SET( HEADERS
include/PVRTCCompressor.h include/PVRTCCompressor.h
src/Block.h src/Block.h
src/Image.h src/PVRTCImage.h
) )
SET( SOURCES SET( SOURCES
src/Compressor.cpp src/Compressor.cpp
src/Decompressor.cpp src/Decompressor.cpp
src/Block.cpp src/Block.cpp
src/Image.cpp src/PVRTCImage.cpp
) )
ADD_LIBRARY( PVRTCEncoder ADD_LIBRARY( PVRTCEncoder

View file

@ -58,7 +58,7 @@
#include <vector> #include <vector>
#include "Pixel.h" #include "Pixel.h"
#include "Image.h" #include "PVRTCImage.h"
#include "Block.h" #include "Block.h"
namespace PVRTCC { namespace PVRTCC {

View file

@ -57,7 +57,7 @@
#include "Pixel.h" #include "Pixel.h"
#include "Block.h" #include "Block.h"
#include "Image.h" #include "PVRTCImage.h"
namespace PVRTCC { namespace PVRTCC {
@ -94,7 +94,7 @@ namespace PVRTCC {
assert(imgA.GetWidth() == imgB.GetWidth()); assert(imgA.GetWidth() == imgB.GetWidth());
assert(imgA.GetHeight() == imgB.GetHeight()); assert(imgA.GetHeight() == imgB.GetHeight());
Image debugModulation(h, w); Image debugModulation(w, h);
const uint8 debugModulationBitDepth[4] = { 8, 4, 4, 4 }; const uint8 debugModulationBitDepth[4] = { 8, 4, 4, 4 };
debugModulation.ChangeBitDepth(debugModulationBitDepth); debugModulation.ChangeBitDepth(debugModulationBitDepth);
@ -113,7 +113,6 @@ namespace PVRTCC {
const Pixel &pa = imgA(i, j); const Pixel &pa = imgA(i, j);
const Pixel &pb = imgB(i, j); const Pixel &pb = imgB(i, j);
Pixel result;
bool punchThrough = false; bool punchThrough = false;
uint8 lerpVal = 0; uint8 lerpVal = 0;
if(b.GetModeBit()) { if(b.GetModeBit()) {
@ -147,6 +146,7 @@ namespace PVRTCC {
} }
} }
Pixel result;
for(uint32 c = 0; c < 4; c++) { for(uint32 c = 0; c < 4; c++) {
uint16 va = static_cast<uint16>(pa.Component(c)); uint16 va = static_cast<uint16>(pa.Component(c));
uint16 vb = static_cast<uint16>(pb.Component(c)); uint16 vb = static_cast<uint16>(pb.Component(c));
@ -274,7 +274,7 @@ namespace PVRTCC {
} }
if(bDebugImages) { if(bDebugImages) {
Image dbgMod(h, w); Image dbgMod(w, h);
for(uint32 i = 0; i < h*w; i++) { for(uint32 i = 0; i < h*w; i++) {
float fb = static_cast<float>(modValues[i]); float fb = static_cast<float>(modValues[i]);
uint8 val = static_cast<uint8>((fb / 8.0f) * 15.0f); uint8 val = static_cast<uint8>((fb / 8.0f) * 15.0f);
@ -324,8 +324,8 @@ namespace PVRTCC {
assert(blocks.size() > 0); assert(blocks.size() > 0);
// Extract the endpoints into A and B images // Extract the endpoints into A and B images
Image imgA(blocksH, blocksW); Image imgA(blocksW, blocksH);
Image imgB(blocksH, blocksW); Image imgB(blocksW, blocksH);
for(uint32 j = 0; j < blocksH; j++) { for(uint32 j = 0; j < blocksH; j++) {
for(uint32 i = 0; i < blocksW; i++) { for(uint32 i = 0; i < blocksW; i++) {

View file

@ -55,7 +55,7 @@
# define snprintf _snprintf # define snprintf _snprintf
#endif #endif
#include "Image.h" #include "PVRTCImage.h"
#include <algorithm> #include <algorithm>
#include <cassert> #include <cassert>
@ -80,55 +80,39 @@ static float ConvertChannelToFloat(uint8 channel, uint8 bitDepth) {
namespace PVRTCC { namespace PVRTCC {
Image::Image(uint32 height, uint32 width) Image::Image(uint32 width, uint32 height)
: m_Width(width) : FasTC::Image<FasTC::Pixel>(width, height)
, m_Height(height) , m_FractionalPixels(new FasTC::Pixel[width * height]) {
, m_Pixels(new Pixel[width * height])
, m_FractionalPixels(new Pixel[width * height]) {
assert(width > 0); assert(width > 0);
assert(height > 0); assert(height > 0);
} }
Image::Image(uint32 height, uint32 width, const Pixel *pixels) Image::Image(uint32 width, uint32 height, const FasTC::Pixel *pixels)
: m_Width(width) : FasTC::Image<FasTC::Pixel>(width, height, pixels)
, m_Height(height) , m_FractionalPixels(new FasTC::Pixel[width * height]) {
, m_Pixels(new Pixel[width * height])
, m_FractionalPixels(new Pixel[width * height]) {
assert(width > 0); assert(width > 0);
assert(height > 0); assert(height > 0);
memcpy(m_Pixels, pixels, width * height * sizeof(Pixel));
} }
Image::Image(const Image &other) Image::Image(const Image &other)
: m_Width(other.GetWidth()) : FasTC::Image<FasTC::Pixel>(other)
, m_Height(other.GetHeight()) , m_FractionalPixels(new FasTC::Pixel[other.GetWidth() * other.GetHeight()]) {
, m_Pixels(new Pixel[other.GetWidth() * other.GetHeight()]) memcpy(m_FractionalPixels, other.m_FractionalPixels, GetWidth() * GetHeight() * sizeof(FasTC::Pixel));
, m_FractionalPixels(new Pixel[other.GetWidth() * other.GetHeight()]) {
memcpy(m_Pixels, other.m_Pixels, GetWidth() * GetHeight() * sizeof(Pixel));
} }
Image &Image::operator=(const Image &other) { Image &Image::operator=(const Image &other) {
m_Width = other.GetWidth(); FasTC::Image<FasTC::Pixel>::operator=(other);
m_Height = other.GetHeight();
assert(m_Pixels);
delete m_Pixels;
m_Pixels = new Pixel[other.GetWidth() * other.GetHeight()];
memcpy(m_Pixels, other.m_Pixels, GetWidth() * GetHeight() * sizeof(Pixel));
assert(m_FractionalPixels); assert(m_FractionalPixels);
delete m_FractionalPixels; delete m_FractionalPixels;
m_FractionalPixels = new Pixel[other.GetWidth() * other.GetHeight()]; m_FractionalPixels = new FasTC::Pixel[other.GetWidth() * other.GetHeight()];
memcpy(m_FractionalPixels, other.m_FractionalPixels, memcpy(m_FractionalPixels, other.m_FractionalPixels,
GetWidth() * GetHeight() * sizeof(Pixel)); GetWidth() * GetHeight() * sizeof(FasTC::Pixel));
return *this; return *this;
} }
Image::~Image() { Image::~Image() {
assert(m_Pixels);
delete [] m_Pixels;
assert(m_FractionalPixels); assert(m_FractionalPixels);
delete [] m_FractionalPixels; delete [] m_FractionalPixels;
} }
@ -155,18 +139,18 @@ void Image::BilinearUpscale(uint32 xtimes, uint32 ytimes,
const uint32 yscale = 1 << ytimes; const uint32 yscale = 1 << ytimes;
const uint32 yoffset = yscale >> 1; const uint32 yoffset = yscale >> 1;
Pixel *upscaledPixels = new Pixel[newWidth * newHeight]; FasTC::Pixel *upscaledPixels = new FasTC::Pixel[newWidth * newHeight];
assert(m_FractionalPixels); assert(m_FractionalPixels);
delete m_FractionalPixels; delete m_FractionalPixels;
m_FractionalPixels = new Pixel[newWidth * newHeight]; m_FractionalPixels = new FasTC::Pixel[newWidth * newHeight];
for(uint32 j = 0; j < newHeight; j++) { for(uint32 j = 0; j < newHeight; j++) {
for(uint32 i = 0; i < newWidth; i++) { for(uint32 i = 0; i < newWidth; i++) {
const uint32 pidx = j * newWidth + i; const uint32 pidx = j * newWidth + i;
Pixel &p = upscaledPixels[pidx]; FasTC::Pixel &p = upscaledPixels[pidx];
Pixel &fp = m_FractionalPixels[pidx]; FasTC::Pixel &fp = m_FractionalPixels[pidx];
const int32 highXIdx = (i + xoffset) / xscale; const int32 highXIdx = (i + xoffset) / xscale;
const int32 lowXIdx = highXIdx - 1; const int32 lowXIdx = highXIdx - 1;
@ -183,10 +167,10 @@ void Image::BilinearUpscale(uint32 xtimes, uint32 ytimes,
const uint32 bottomLeftWeight = lowXWeight * highYWeight; const uint32 bottomLeftWeight = lowXWeight * highYWeight;
const uint32 bottomRightWeight = highXWeight * highYWeight; const uint32 bottomRightWeight = highXWeight * highYWeight;
const Pixel &topLeft = GetPixel(lowXIdx, lowYIdx, wrapMode); const FasTC::Pixel &topLeft = GetPixel(lowXIdx, lowYIdx, wrapMode);
const Pixel &topRight = GetPixel(highXIdx, lowYIdx, wrapMode); const FasTC::Pixel &topRight = GetPixel(highXIdx, lowYIdx, wrapMode);
const Pixel &bottomLeft = GetPixel(lowXIdx, highYIdx, wrapMode); const FasTC::Pixel &bottomLeft = GetPixel(lowXIdx, highYIdx, wrapMode);
const Pixel &bottomRight = GetPixel(highXIdx, highYIdx, wrapMode); const FasTC::Pixel &bottomRight = GetPixel(highXIdx, highYIdx, wrapMode);
// Make sure the bit depth matches the original... // Make sure the bit depth matches the original...
uint8 bitDepth[4]; uint8 bitDepth[4];
@ -224,10 +208,7 @@ void Image::BilinearUpscale(uint32 xtimes, uint32 ytimes,
} }
} }
delete m_Pixels; SetImageData(newWidth, newHeight, upscaledPixels);
m_Pixels = upscaledPixels;
m_Width = newWidth;
m_Height = newHeight;
} }
void Image::ContentAwareDownscale(uint32 xtimes, uint32 ytimes, void Image::ContentAwareDownscale(uint32 xtimes, uint32 ytimes,
@ -238,11 +219,11 @@ void Image::ContentAwareDownscale(uint32 xtimes, uint32 ytimes,
const uint32 newWidth = w >> xtimes; const uint32 newWidth = w >> xtimes;
const uint32 newHeight = h >> ytimes; const uint32 newHeight = h >> ytimes;
Pixel *downscaledPixels = new Pixel[newWidth * newHeight]; FasTC::Pixel *downscaledPixels = new FasTC::Pixel[newWidth * newHeight];
const uint32 numDownscaledPixels = newWidth * newHeight; const uint32 numDownscaledPixels = newWidth * newHeight;
uint8 bitDepth[4]; uint8 bitDepth[4];
m_Pixels[0].GetBitDepth(bitDepth); GetPixels()[0].GetBitDepth(bitDepth);
for(uint32 i = 0; i < numDownscaledPixels; i++) { for(uint32 i = 0; i < numDownscaledPixels; i++) {
downscaledPixels[i].ChangeBitDepth(bitDepth); downscaledPixels[i].ChangeBitDepth(bitDepth);
@ -282,10 +263,10 @@ void Image::ContentAwareDownscale(uint32 xtimes, uint32 ytimes,
for(uint32 i = 0; i < w * h; i++) { for(uint32 i = 0; i < w * h; i++) {
// First convert the pixel values to floats using // First convert the pixel values to floats using
// premultiplied alpha... // premultiplied alpha...
float a = ConvertChannelToFloat(m_Pixels[i].A(), bitDepth[0]); float a = ConvertChannelToFloat(GetPixels()[i].A(), bitDepth[0]);
float r = a * ConvertChannelToFloat(m_Pixels[i].R(), bitDepth[1]); float r = a * ConvertChannelToFloat(GetPixels()[i].R(), bitDepth[1]);
float g = a * ConvertChannelToFloat(m_Pixels[i].G(), bitDepth[2]); float g = a * ConvertChannelToFloat(GetPixels()[i].G(), bitDepth[2]);
float b = a * ConvertChannelToFloat(m_Pixels[i].B(), bitDepth[3]); float b = a * ConvertChannelToFloat(GetPixels()[i].B(), bitDepth[3]);
I[i] = r * 0.21f + g * 0.71f + b * 0.07f; I[i] = r * 0.21f + g * 0.71f + b * 0.07f;
} }
@ -308,7 +289,7 @@ void Image::ContentAwareDownscale(uint32 xtimes, uint32 ytimes,
Iy[idx] = (I[yphidx] - I[ymhidx]) / 2.0f; Iy[idx] = (I[yphidx] - I[ymhidx]) / 2.0f;
for(uint32 c = 0; c <= 3; c++) { for(uint32 c = 0; c <= 3; c++) {
#define CPNT(dx) ConvertChannelToFloat(m_Pixels[dx].Component(c), bitDepth[c]) #define CPNT(dx) ConvertChannelToFloat(GetPixels()[dx].Component(c), bitDepth[c])
Ix[c][idx] = (CPNT(xphidx) - CPNT(xmhidx)) / 2.0f; Ix[c][idx] = (CPNT(xphidx) - CPNT(xmhidx)) / 2.0f;
Ixx[c][idx] = (CPNT(xphidx) - 2.0f*CPNT(idx) + CPNT(xmhidx)) / 2.0f; Ixx[c][idx] = (CPNT(xphidx) - 2.0f*CPNT(idx) + CPNT(xmhidx)) / 2.0f;
Iyy[c][idx] = (CPNT(yphidx) - 2.0f*CPNT(idx) + CPNT(ymhidx)) / 2.0f; Iyy[c][idx] = (CPNT(yphidx) - 2.0f*CPNT(idx) + CPNT(ymhidx)) / 2.0f;
@ -339,9 +320,9 @@ void Image::ContentAwareDownscale(uint32 xtimes, uint32 ytimes,
} }
uint32 idx = GetPixelIndex(x, y); uint32 idx = GetPixelIndex(x, y);
Pixel current = m_Pixels[idx]; FasTC::Pixel current = GetPixels()[idx];
Pixel result; FasTC::Pixel result;
result.ChangeBitDepth(bitDepth); result.ChangeBitDepth(bitDepth);
float Ixsq = Ix[4][idx] * Ix[4][idx]; float Ixsq = Ix[4][idx] * Ix[4][idx];
@ -364,26 +345,21 @@ void Image::ContentAwareDownscale(uint32 xtimes, uint32 ytimes,
} }
} }
delete m_Pixels; SetImageData(newWidth, newHeight, downscaledPixels);
m_Pixels = downscaledPixels;
m_Width = newWidth;
m_Height = newHeight;
delete [] imgData; delete [] imgData;
} }
void Image::ChangeBitDepth(const uint8 (&depths)[4]) { void Image::ChangeBitDepth(const uint8 (&depths)[4]) {
for(uint32 j = 0; j < GetHeight(); j++) { for(uint32 j = 0; j < GetHeight(); j++) {
for(uint32 i = 0; i < GetWidth(); i++) { for(uint32 i = 0; i < GetWidth(); i++) {
uint32 pidx = j * GetWidth() + i; (*this)(i, j).ChangeBitDepth(depths);
m_Pixels[pidx].ChangeBitDepth(depths);
} }
} }
} }
void Image::ExpandTo8888() { void Image::ExpandTo8888() {
uint8 currentDepth[4]; uint8 currentDepth[4];
m_Pixels[0].GetBitDepth(currentDepth); GetPixels()[0].GetBitDepth(currentDepth);
uint8 fractionDepth[4]; uint8 fractionDepth[4];
const uint8 fullDepth[4] = { 8, 8, 8, 8 }; const uint8 fullDepth[4] = { 8, 8, 8, 8 };
@ -391,8 +367,10 @@ void Image::ExpandTo8888() {
for(uint32 j = 0; j < GetHeight(); j++) { for(uint32 j = 0; j < GetHeight(); j++) {
for(uint32 i = 0; i < GetWidth(); i++) { for(uint32 i = 0; i < GetWidth(); i++) {
FasTC::Pixel &p = (*this)(i, j);
p.ChangeBitDepth(fullDepth);
uint32 pidx = j * GetWidth() + i; uint32 pidx = j * GetWidth() + i;
m_Pixels[pidx].ChangeBitDepth(fullDepth);
m_FractionalPixels[pidx].GetBitDepth(fractionDepth); m_FractionalPixels[pidx].GetBitDepth(fractionDepth);
for(uint32 c = 0; c < 4; c++) { for(uint32 c = 0; c < 4; c++) {
@ -402,17 +380,17 @@ void Image::ExpandTo8888() {
uint32 shift = fractionDepth[c] - (fullDepth[c] - currentDepth[c]); uint32 shift = fractionDepth[c] - (fullDepth[c] - currentDepth[c]);
uint32 fractionBits = m_FractionalPixels[pidx].Component(c) >> shift; uint32 fractionBits = m_FractionalPixels[pidx].Component(c) >> shift;
uint32 component = m_Pixels[pidx].Component(c); uint32 component = p.Component(c);
component += ((fractionBits * numerator) / denominator); component += ((fractionBits * numerator) / denominator);
m_Pixels[pidx].Component(c) = component; p.Component(c) = component;
} }
} }
} }
} }
const Pixel &Image::GetPixel(int32 i, int32 j, EWrapMode wrapMode) const { const FasTC::Pixel &Image::GetPixel(int32 i, int32 j, EWrapMode wrapMode) const {
return m_Pixels[GetPixelIndex(i, j, wrapMode)]; return GetPixels()[GetPixelIndex(i, j, wrapMode)];
} }
const uint32 Image::GetPixelIndex(int32 i, int32 j, EWrapMode wrapMode) const { const uint32 Image::GetPixelIndex(int32 i, int32 j, EWrapMode wrapMode) const {
@ -454,33 +432,20 @@ const uint32 Image::GetPixelIndex(int32 i, int32 j, EWrapMode wrapMode) const {
return idx; return idx;
} }
Pixel & Image::operator()(uint32 i, uint32 j) {
assert(i < GetWidth());
assert(j < GetHeight());
return m_Pixels[j * GetWidth() + i];
}
const Pixel & Image::operator()(uint32 i, uint32 j) const {
assert(i < GetWidth());
assert(j < GetHeight());
return m_Pixels[j * GetWidth() + i];
}
void Image::DebugOutput(const char *filename) const { void Image::DebugOutput(const char *filename) const {
uint32 *outPixels = new uint32[GetWidth() * GetHeight()]; uint32 *outPixels = new uint32[GetWidth() * GetHeight()];
const uint8 fullDepth[4] = { 8, 8, 8, 8 }; const uint8 fullDepth[4] = { 8, 8, 8, 8 };
for(uint32 j = 0; j < GetHeight(); j++) { for(uint32 j = 0; j < GetHeight(); j++) {
for(uint32 i = 0; i < GetWidth(); i++) { for(uint32 i = 0; i < GetWidth(); i++) {
uint32 idx = j * GetWidth() + i; FasTC::Pixel p = (*this)(i, j);
Pixel p = m_Pixels[idx];
p.ChangeBitDepth(fullDepth); p.ChangeBitDepth(fullDepth);
p.A() = 255; p.A() = 255;
outPixels[idx] = p.Pack(); outPixels[j*GetWidth() + i] = p.Pack();
} }
} }
::Image img(GetWidth(), GetHeight(), outPixels); FasTC::Image<> img(GetWidth(), GetHeight(), outPixels);
char debugFilename[256]; char debugFilename[256];
snprintf(debugFilename, sizeof(debugFilename), "%s.png", filename); snprintf(debugFilename, sizeof(debugFilename), "%s.png", filename);

View file

@ -55,6 +55,7 @@
#include "TexCompTypes.h" #include "TexCompTypes.h"
#include "PVRTCCompressor.h" #include "PVRTCCompressor.h"
#include "Image.h"
// Forward include // Forward include
namespace FasTC { namespace FasTC {
@ -63,15 +64,13 @@ namespace FasTC {
namespace PVRTCC { namespace PVRTCC {
using FasTC::Pixel; class Image : public FasTC::Image<FasTC::Pixel> {
class Image {
public: public:
Image(uint32 height, uint32 width); Image(uint32 width, uint32 height);
Image(uint32 height, uint32 width, const Pixel *pixels); Image(uint32 width, uint32 height, const FasTC::Pixel *pixels);
Image(const Image &); Image(const Image &);
Image &operator=(const Image &); Image &operator=(const Image &);
~Image(); virtual ~Image();
void BilinearUpscale(uint32 xtimes, uint32 ytimes, void BilinearUpscale(uint32 xtimes, uint32 ytimes,
EWrapMode wrapMode = eWrapMode_Wrap); EWrapMode wrapMode = eWrapMode_Wrap);
@ -84,25 +83,15 @@ class Image {
EWrapMode wrapMode = eWrapMode_Wrap, EWrapMode wrapMode = eWrapMode_Wrap,
bool bOffsetNewPixels = false); bool bOffsetNewPixels = false);
void ChangeBitDepth(const uint8 (&depths)[4]);
void ExpandTo8888(); void ExpandTo8888();
void ChangeBitDepth(const uint8 (&depths)[4]);
Pixel &operator()(uint32 i, uint32 j);
const Pixel &operator()(uint32 i, uint32 j) const;
uint32 GetWidth() const { return m_Width; }
uint32 GetHeight() const { return m_Height; }
void DebugOutput(const char *filename) const; void DebugOutput(const char *filename) const;
private: private:
uint32 m_Width; FasTC::Pixel *m_FractionalPixels;
uint32 m_Height;
Pixel *m_Pixels;
Pixel *m_FractionalPixels;
const uint32 GetPixelIndex(int32 i, int32 j, EWrapMode wrapMode = eWrapMode_Clamp) const; const uint32 GetPixelIndex(int32 i, int32 j, EWrapMode wrapMode = eWrapMode_Clamp) const;
const Pixel &GetPixel(int32 i, int32 j, EWrapMode wrapMode = eWrapMode_Clamp) const; const FasTC::Pixel &GetPixel(int32 i, int32 j, EWrapMode wrapMode = eWrapMode_Clamp) const;
}; };
} // namespace PVRTCC } // namespace PVRTCC

View file

@ -60,7 +60,7 @@ SET(TESTS
) )
FOREACH(TEST ${TESTS}) FOREACH(TEST ${TESTS})
SET(TEST_NAME Test${TEST}) SET(TEST_NAME Test_PVRTCEncoder_${TEST})
SET(TEST_MODULE ${TEST}Test.cpp) SET(TEST_MODULE ${TEST}Test.cpp)
ADD_EXECUTABLE(${TEST_NAME} ${TEST_MODULE}) ADD_EXECUTABLE(${TEST_NAME} ${TEST_MODULE})
TARGET_LINK_LIBRARIES(${TEST_NAME} PVRTCEncoder) TARGET_LINK_LIBRARIES(${TEST_NAME} PVRTCEncoder)
@ -71,7 +71,7 @@ ENDFOREACH()
# Test the decompressor against the included PVR Texture library.... # Test the decompressor against the included PVR Texture library....
IF(PVRTEXLIB_FOUND) IF(PVRTEXLIB_FOUND)
SET(TEST_NAME TestDecompVersusPVRLib) SET(TEST_NAME Test_PVRTCEncoder_DecompVersusPVRLib)
# Copy the .pvr files that we will use for testing... # Copy the .pvr files that we will use for testing...
SET(TEST_IMAGES SET(TEST_IMAGES

View file

@ -51,78 +51,14 @@
*/ */
#include "gtest/gtest.h" #include "gtest/gtest.h"
#include "Image.h" #include "PVRTCImage.h"
#include "Pixel.h" #include "Pixel.h"
#include "TestUtils.h" #include "TestUtils.h"
#include <cstdlib> #include <cstdlib>
TEST(Image, NonSpecificConstructor) {
PVRTCC::Pixel p;
PVRTCC::Image img (4, 4);
for(uint32 i = 0; i < 4; i++) {
for(uint32 j = 0; j < 4; j++) {
EXPECT_TRUE(img(i, j) == p);
}
}
}
TEST(Image, SpecificConstructor) {
PVRTCC::Pixel pxs[16];
for(uint32 i = 0; i < 4; i++) {
for(uint32 j = 0; j < 4; j++) {
pxs[j*4 + i].R() = i;
pxs[j*4 + i].G() = j;
}
}
PVRTCC::Image img(4, 4, pxs);
for(uint32 i = 0; i < 4; i++) {
for(uint32 j = 0; j < 4; j++) {
EXPECT_TRUE(img(i, j) == pxs[j*4 + i]);
}
}
}
TEST(Image, CopyConstructor) {
PVRTCC::Pixel pxs[16];
for(uint32 i = 0; i < 4; i++) {
for(uint32 j = 0; j < 4; j++) {
pxs[j*4 + i].R() = i;
pxs[j*4 + i].G() = j;
}
}
PVRTCC::Image img(4, 4, pxs);
PVRTCC::Image img2(img);
for(uint32 i = 0; i < 4; i++) {
for(uint32 j = 0; j < 4; j++) {
EXPECT_TRUE(img2(i, j) == pxs[j*4 + i]);
}
}
}
TEST(Image, AssignmentOperator) {
PVRTCC::Pixel pxs[16];
for(uint32 i = 0; i < 4; i++) {
for(uint32 j = 0; j < 4; j++) {
pxs[j*4 + i].R() = i;
pxs[j*4 + i].G() = j;
}
}
PVRTCC::Image img(4, 4, pxs);
PVRTCC::Image img2 = img;
for(uint32 i = 0; i < 4; i++) {
for(uint32 j = 0; j < 4; j++) {
EXPECT_TRUE(img2(i, j) == pxs[j*4 + i]);
}
}
}
TEST(Image, BilinearUpscale) { TEST(Image, BilinearUpscale) {
PVRTCC::Pixel pxs[16]; FasTC::Pixel pxs[16];
for(uint32 i = 0; i < 4; i++) { for(uint32 i = 0; i < 4; i++) {
for(uint32 j = 0; j < 4; j++) { for(uint32 j = 0; j < 4; j++) {
pxs[j*4 + i].R() = i*2; pxs[j*4 + i].R() = i*2;
@ -152,7 +88,6 @@ TEST(Image, BilinearUpscale) {
} }
} }
TEST(Image, BilinearUpscaleMaintainsPixels) { TEST(Image, BilinearUpscaleMaintainsPixels) {
srand(0xabd1ca7e); srand(0xabd1ca7e);
@ -160,7 +95,7 @@ TEST(Image, BilinearUpscaleMaintainsPixels) {
const uint32 w = 4; const uint32 w = 4;
const uint32 h = 4; const uint32 h = 4;
PVRTCC::Pixel pxs[16]; FasTC::Pixel pxs[16];
for(uint32 i = 0; i < w; i++) { for(uint32 i = 0; i < w; i++) {
for(uint32 j = 0; j < h; j++) { for(uint32 j = 0; j < h; j++) {
pxs[j*w + i].R() = rand() % 256; pxs[j*w + i].R() = rand() % 256;
@ -177,7 +112,7 @@ TEST(Image, BilinearUpscaleMaintainsPixels) {
for(uint32 i = 2; i < img.GetWidth(); i+=4) { for(uint32 i = 2; i < img.GetWidth(); i+=4) {
for(uint32 j = 2; j < img.GetHeight(); j+=4) { for(uint32 j = 2; j < img.GetHeight(); j+=4) {
PVRTCC::Pixel p = img(i, j); FasTC::Pixel p = img(i, j);
uint32 idx = ((j - 2) / 4) * w + ((i-2)/4); uint32 idx = ((j - 2) / 4) * w + ((i-2)/4);
EXPECT_EQ(PixelPrinter(p.Pack()), PixelPrinter(pxs[idx].Pack())); EXPECT_EQ(PixelPrinter(p.Pack()), PixelPrinter(pxs[idx].Pack()));
} }
@ -190,7 +125,7 @@ TEST(Image, NonuniformBilinearUpscale) {
const uint32 kWidth = 4; const uint32 kWidth = 4;
const uint32 kHeight = 8; const uint32 kHeight = 8;
PVRTCC::Pixel pxs[kWidth * kHeight]; FasTC::Pixel pxs[kWidth * kHeight];
for(uint32 i = 0; i < kWidth; i++) { for(uint32 i = 0; i < kWidth; i++) {
for(uint32 j = 0; j < kHeight; j++) { for(uint32 j = 0; j < kHeight; j++) {
pxs[j*kWidth + i].R() = i*4; pxs[j*kWidth + i].R() = i*4;
@ -198,7 +133,7 @@ TEST(Image, NonuniformBilinearUpscale) {
} }
} }
PVRTCC::Image img(kHeight, kWidth, pxs); PVRTCC::Image img(kWidth, kHeight, pxs);
img.BilinearUpscale(2, 1, PVRTCC::eWrapMode_Clamp); img.BilinearUpscale(2, 1, PVRTCC::eWrapMode_Clamp);
EXPECT_EQ(img.GetWidth(), static_cast<uint32>(kWidth << 2)); EXPECT_EQ(img.GetWidth(), static_cast<uint32>(kWidth << 2));
EXPECT_EQ(img.GetHeight(), static_cast<uint32>(kHeight << 1)); EXPECT_EQ(img.GetHeight(), static_cast<uint32>(kHeight << 1));
@ -223,7 +158,7 @@ TEST(Image, NonuniformBilinearUpscale) {
} }
TEST(Image, BilinearUpscaleWrapped) { TEST(Image, BilinearUpscaleWrapped) {
PVRTCC::Pixel pxs[16]; FasTC::Pixel pxs[16];
// Make sure that our bit depth is less than full... // Make sure that our bit depth is less than full...
for(uint32 i = 0; i < 16; i++) { for(uint32 i = 0; i < 16; i++) {
@ -245,7 +180,7 @@ TEST(Image, BilinearUpscaleWrapped) {
for(uint32 i = 0; i < img.GetWidth(); i++) { for(uint32 i = 0; i < img.GetWidth(); i++) {
for(uint32 j = 0; j < img.GetHeight(); j++) { for(uint32 j = 0; j < img.GetHeight(); j++) {
const PVRTCC::Pixel &p = img(i, j); const FasTC::Pixel &p = img(i, j);
// First make sure that the bit depth didn't change // First make sure that the bit depth didn't change
uint8 depth[4]; uint8 depth[4];
@ -284,9 +219,9 @@ TEST(Image, ContentAwareDownscale) {
for(uint32 j = 0; j < img.GetHeight(); j++) { for(uint32 j = 0; j < img.GetHeight(); j++) {
for(uint32 i = 0; i < img.GetWidth(); i++) { for(uint32 i = 0; i < img.GetWidth(); i++) {
if(j < 4) { if(j < 4) {
img(i, j) = PVRTCC::Pixel( 0xFF000000 ); img(i, j) = FasTC::Pixel( 0xFF000000 );
} else { } else {
img(i, j) = PVRTCC::Pixel( 0xFF0000FF ); img(i, j) = FasTC::Pixel( 0xFF0000FF );
} }
} }
} }