Add stats function to serial encoder. Otherwise, continue to use non-stats function in threaded mode.

This commit is contained in:
Pavel Krajcevski 2012-10-07 00:34:06 -04:00
parent 1e6a2d4c7b
commit 99534bc5d0
6 changed files with 98 additions and 15 deletions

View file

@ -1466,7 +1466,7 @@ namespace BC7C
} }
} }
void CompressImageBC7( void CompressImageBC7Stats(
const unsigned char *inBuf, const unsigned char *inBuf,
unsigned char *outBuf, unsigned char *outBuf,
unsigned int width, unsigned int width,
@ -1655,6 +1655,10 @@ namespace BC7C
uint32 blockIdx = 0; uint32 blockIdx = 0;
if(statManager) { if(statManager) {
blockIdx = statManager->BeginBlock(); blockIdx = statManager->BeginBlock();
for(int i = 0; i < kNumBlockStats; i++) {
statManager->AddStat(blockIdx, BlockStat(kBlockStatString[i], 0));
}
} }
// All a single color? // All a single color?

View file

@ -2,6 +2,7 @@
#include <stdio.h> #include <stdio.h>
#include <string.h> #include <string.h>
#include "BlockStats.h"
#include "TexComp.h" #include "TexComp.h"
#include "ImageFile.h" #include "ImageFile.h"
#include "Image.h" #include "Image.h"
@ -99,7 +100,12 @@ int main(int argc, char **argv) {
if(NULL == img) { if(NULL == img) {
fprintf(stderr, "Error loading file: %s\n", argv[fileArg]); fprintf(stderr, "Error loading file: %s\n", argv[fileArg]);
return 1; return 1;
} }
const Image *img = file.GetImage();
int numBlocks = (img->GetWidth() * img->GetHeight())/16;
BlockStatManager *statManager = new BlockStatManager(numBlocks);
SCompressionSettings settings; SCompressionSettings settings;
settings.bUseSIMD = bUseSIMD; settings.bUseSIMD = bUseSIMD;
@ -107,6 +113,7 @@ int main(int argc, char **argv) {
settings.iQuality = quality; settings.iQuality = quality;
settings.iNumCompressions = numCompressions; settings.iNumCompressions = numCompressions;
settings.iJobSize = numJobs; settings.iJobSize = numJobs;
settings.pStatManager = statManager;
CompressedImage *ci = img->Compress(settings); CompressedImage *ci = img->Compress(settings);
if(NULL == ci) { if(NULL == ci) {
@ -122,8 +129,11 @@ int main(int argc, char **argv) {
fprintf(stderr, "Error computing PSNR\n"); fprintf(stderr, "Error computing PSNR\n");
} }
statManager->ToFile(strcat(argv[fileArg], "-log.txt"));
// Cleanup // Cleanup
delete ci; delete ci;
delete statManager;
return 0; return 0;
} }

View file

@ -16,7 +16,7 @@ public:
private: private:
static const int kStatNameSz = 32; static const int kStatNameSz = 32;
char m_StatName[kStatNameSz]; CHAR m_StatName[kStatNameSz];
union { union {
uint64 m_IntStat; uint64 m_IntStat;
double m_FloatStat; double m_FloatStat;
@ -31,6 +31,7 @@ class BlockStatManager {
uint32 BeginBlock(); uint32 BeginBlock();
void AddStat(uint32 blockIdx, const BlockStat &stat); void AddStat(uint32 blockIdx, const BlockStat &stat);
void ToFile(const CHAR *filename);
private: private:
@ -40,6 +41,8 @@ class BlockStatManager {
~BlockStatList(); ~BlockStatList();
void AddStat(const BlockStat &stat); void AddStat(const BlockStat &stat);
BlockStat GetStat() const { return m_Stat; }
const BlockStatList *GetTail() const { return m_Tail; }
private: private:
BlockStatList(const BlockStat &stat); BlockStatList(const BlockStat &stat);

View file

@ -6,6 +6,7 @@
// Forward declarations // Forward declarations
class ImageFile; class ImageFile;
class CompressedImage; class CompressedImage;
class BlockStatManager;
struct SCompressionSettings { struct SCompressionSettings {
SCompressionSettings(); // defaults SCompressionSettings(); // defaults
@ -34,6 +35,12 @@ struct SCompressionSettings {
// number of threads, and each thread will do it's job and // number of threads, and each thread will do it's job and
// exit. // exit.
int iJobSize; int iJobSize;
// This is an optinal pointer to a stat manager class. If
// instantiated and the proper function pointer is defined
// then the compression routine that collects the stats will
// be called.
BlockStatManager *pStatManager;
}; };
extern bool CompressImageData( extern bool CompressImageData(
@ -55,6 +62,18 @@ typedef void (* CompressionFunc)(
unsigned int height // Image height unsigned int height // Image height
); );
// A compression function format. It takes the raw data and image dimensions and
// returns the compressed image data into outData. It is assumed that there is
// enough space allocated for outData to store the compressed data. Allocation
// is dependent on the compression format.
typedef void (* CompressionFuncWithStats)(
const unsigned char *inData, // Raw image data
unsigned char *outData, // Buffer to store compressed data.
unsigned int width, // Image width
unsigned int height, // Image height
BlockStatManager &statManager// Stat manager
);
// This function computes the Peak Signal to Noise Ratio between a // This function computes the Peak Signal to Noise Ratio between a
// compressed image and a raw image. // compressed image and a raw image.
extern double ComputePSNR(const CompressedImage &ci, const ImageFile &file); extern double ComputePSNR(const CompressedImage &ci, const ImageFile &file);

View file

@ -5,6 +5,8 @@
#include <string.h> #include <string.h>
#include <assert.h> #include <assert.h>
#include "FileStream.h"
template <typename T> template <typename T>
static T max(const T &a, const T &b) { static T max(const T &a, const T &b) {
return (a > b)? a : b; return (a > b)? a : b;
@ -59,10 +61,10 @@ BlockStatManager::~BlockStatManager() {
} }
uint32 BlockStatManager::BeginBlock() { uint32 BlockStatManager::BeginBlock() {
if((m_NextBlock + 1) == m_BlockStatListSz) { if(m_NextBlock == m_BlockStatListSz) {
fprintf(stderr, "WARNING -- BlockStatManager::BeginBlock(), reached end of block list.\n"); fprintf(stderr, "WARNING -- BlockStatManager::BeginBlock(), reached end of block list.\n");
assert(false); assert(false);
return m_NextBlock; return m_NextBlock-1;
} }
TCLock lock(m_Mutex); TCLock lock(m_Mutex);
@ -108,7 +110,7 @@ void BlockStatManager::BlockStatList::AddStat(const BlockStat &stat) {
if(strncmp(stat.m_StatName, m_Stat.m_StatName, BlockStat::kStatNameSz) == 0) { if(strncmp(stat.m_StatName, m_Stat.m_StatName, BlockStat::kStatNameSz) == 0) {
m_Stat = stat; m_Stat = stat;
} }
else if(!m_Tail) { else if(m_Tail) {
m_Tail->AddStat(stat); m_Tail->AddStat(stat);
} }
else { else {
@ -120,3 +122,28 @@ void BlockStatManager::BlockStatList::AddStat(const BlockStat &stat) {
} }
} }
} }
void BlockStatManager::ToFile(const CHAR *filename) {
FileStream fstr (filename, eFileMode_Write);
for(int i = 0; i < m_BlockStatListSz; i++) {
const BlockStatList *head = &(m_BlockStatList[i]);
while(head) {
BlockStat s = head->GetStat();
CHAR statStr[256];
snprintf(statStr, 256, "%d: %s, %llu, %f\n", i, s.m_StatName, s.m_IntStat, s.m_FloatStat);
int statStrLen = strlen(statStr);
if(statStrLen > 255) {
statStr[255] = '\n';
statStrLen = 255;
}
fstr.Write((uint8 *)statStr, statStrLen);
head = head->GetTail();
}
}
}

View file

@ -43,6 +43,17 @@ SCompressionSettings:: SCompressionSettings()
clamp(iQuality, 0, 256); clamp(iQuality, 0, 256);
} }
static CompressionFuncWithStats ChooseFuncFromSettingsWithStats(const SCompressionSettings &s) {
switch(s.format) {
case eCompressionFormat_BPTC:
{
return BC7C::CompressImageBC7Stats;
}
break;
}
return NULL;
}
static CompressionFunc ChooseFuncFromSettings(const SCompressionSettings &s) { static CompressionFunc ChooseFuncFromSettings(const SCompressionSettings &s) {
switch(s.format) { switch(s.format) {
case eCompressionFormat_BPTC: case eCompressionFormat_BPTC:
@ -54,7 +65,7 @@ static CompressionFunc ChooseFuncFromSettings(const SCompressionSettings &s) {
} }
else { else {
#endif #endif
return BC7C::CompressImageBC7; return BC7C::CompressImageBC7;
#ifdef HAS_SSE_41 #ifdef HAS_SSE_41
} }
#endif #endif
@ -72,9 +83,11 @@ static double CompressImageInSerial(
const unsigned char *imgData, const unsigned char *imgData,
const unsigned int imgDataSz, const unsigned int imgDataSz,
const SCompressionSettings &settings, const SCompressionSettings &settings,
const CompressionFunc f,
unsigned char *outBuf unsigned char *outBuf
) { ) {
CompressionFunc f = ChooseFuncFromSettings(settings);
CompressionFuncWithStats fStats = ChooseFuncFromSettingsWithStats(settings);
double cmpTimeTotal = 0.0; double cmpTimeTotal = 0.0;
for(int i = 0; i < settings.iNumCompressions; i++) { for(int i = 0; i < settings.iNumCompressions; i++) {
@ -84,7 +97,12 @@ static double CompressImageInSerial(
stopWatch.Start(); stopWatch.Start();
// !FIXME! We're assuming that we have 4x4 blocks here... // !FIXME! We're assuming that we have 4x4 blocks here...
(*f)(imgData, outBuf, imgDataSz / 16, 4); if(fStats && settings.pStatManager) {
(*fStats)(imgData, outBuf, imgDataSz / 16, 4, *(settings.pStatManager));
}
else {
(*f)(imgData, outBuf, imgDataSz / 16, 4);
}
stopWatch.Stop(); stopWatch.Stop();
@ -99,10 +117,11 @@ static double CompressImageWithThreads(
const unsigned char *imgData, const unsigned char *imgData,
const unsigned int imgDataSz, const unsigned int imgDataSz,
const SCompressionSettings &settings, const SCompressionSettings &settings,
const CompressionFunc f,
unsigned char *outBuf unsigned char *outBuf
) { ) {
CompressionFunc f = ChooseFuncFromSettings(settings);
ThreadGroup tgrp (settings.iNumThreads, imgData, imgDataSz, f, outBuf); ThreadGroup tgrp (settings.iNumThreads, imgData, imgDataSz, f, outBuf);
if(!(tgrp.PrepareThreads())) { if(!(tgrp.PrepareThreads())) {
assert(!"Thread group failed to prepare threads?!"); assert(!"Thread group failed to prepare threads?!");
@ -121,7 +140,7 @@ static double CompressImageWithThreads(
cmpTimeTotal += tgrp.GetStopWatch().TimeInMilliseconds(); cmpTimeTotal += tgrp.GetStopWatch().TimeInMilliseconds();
} }
tgrp.CleanUpThreads(); tgrp.CleanUpThreads();
double cmpTime = cmpTimeTotal / double(settings.iNumCompressions); double cmpTime = cmpTimeTotal / double(settings.iNumCompressions);
return cmpTime; return cmpTime;
@ -131,9 +150,10 @@ static double CompressImageWithWorkerQueue(
const unsigned char *imgData, const unsigned char *imgData,
const unsigned int imgDataSz, const unsigned int imgDataSz,
const SCompressionSettings &settings, const SCompressionSettings &settings,
const CompressionFunc f,
unsigned char *outBuf unsigned char *outBuf
) { ) {
CompressionFunc f = ChooseFuncFromSettings(settings);
WorkerQueue wq ( WorkerQueue wq (
settings.iNumCompressions, settings.iNumCompressions,
settings.iNumThreads, settings.iNumThreads,
@ -196,12 +216,12 @@ bool CompressImageData(
if(settings.iNumThreads > 1) { if(settings.iNumThreads > 1) {
if(settings.iJobSize > 0) if(settings.iJobSize > 0)
cmpMSTime = CompressImageWithWorkerQueue(data, dataSz, settings, f, cmpData); cmpMSTime = CompressImageWithWorkerQueue(data, dataSz, settings, cmpData);
else else
cmpMSTime = CompressImageWithThreads(data, dataSz, settings, f, cmpData); cmpMSTime = CompressImageWithThreads(data, dataSz, settings, cmpData);
} }
else { else {
cmpMSTime = CompressImageInSerial(data, dataSz, settings, f, cmpData); cmpMSTime = CompressImageInSerial(data, dataSz, settings, cmpData);
} }
// Report compression time // Report compression time