mirror of
https://github.com/yuzu-emu/FasTC.git
synced 2025-01-23 19:11:05 +00:00
Add tests and IDCT... now we need to fix the tests so they pass
This commit is contained in:
parent
4afacf384d
commit
1f2358be79
|
@ -132,6 +132,7 @@ namespace FasTC {
|
||||||
Image<IPixel> *channelThree);
|
Image<IPixel> *channelThree);
|
||||||
|
|
||||||
extern void DiscreteCosineXForm(Image<IPixel> *img, int blockSize);
|
extern void DiscreteCosineXForm(Image<IPixel> *img, int blockSize);
|
||||||
|
extern void InvDiscreteCosineXForm(Image<IPixel> *img, int blockSize);
|
||||||
} // namespace FasTC
|
} // namespace FasTC
|
||||||
|
|
||||||
#endif // __TEXCOMP_IMAGE_H__
|
#endif // __TEXCOMP_IMAGE_H__
|
||||||
|
|
|
@ -48,6 +48,8 @@
|
||||||
#include <cstdio>
|
#include <cstdio>
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
#include <cassert>
|
#include <cassert>
|
||||||
|
|
||||||
|
#define _USE_MATH_DEFINES
|
||||||
#include <cmath>
|
#include <cmath>
|
||||||
|
|
||||||
#include "FasTC/Color.h"
|
#include "FasTC/Color.h"
|
||||||
|
@ -616,6 +618,8 @@ void SplitChannels<Pixel>(const Image<Pixel> &in,
|
||||||
//
|
//
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
typedef void (*DCTBlockFn)(Image<IPixel> *);
|
||||||
|
|
||||||
static void DCT(Image<IPixel> *img) {
|
static void DCT(Image<IPixel> *img) {
|
||||||
Image<IPixel> new_img = *img;
|
Image<IPixel> new_img = *img;
|
||||||
|
|
||||||
|
@ -625,29 +629,74 @@ static void DCT(Image<IPixel> *img) {
|
||||||
for (unsigned int v = 0; v < img->GetHeight(); ++v) {
|
for (unsigned int v = 0; v < img->GetHeight(); ++v) {
|
||||||
for (unsigned int u = 0; u < img->GetWidth(); ++u) {
|
for (unsigned int u = 0; u < img->GetWidth(); ++u) {
|
||||||
new_img(u, v) = 0.0f;
|
new_img(u, v) = 0.0f;
|
||||||
|
float fu = static_cast<float>(u);
|
||||||
|
float fv = static_cast<float>(v);
|
||||||
for (unsigned int y = 0; y < img->GetHeight(); ++y) {
|
for (unsigned int y = 0; y < img->GetHeight(); ++y) {
|
||||||
for (unsigned int x = 0; x < img->GetWidth(); ++x) {
|
for (unsigned int x = 0; x < img->GetWidth(); ++x) {
|
||||||
float fx = static_cast<float>(x);
|
float fx = static_cast<float>(x);
|
||||||
float fy = static_cast<float>(y);
|
float fy = static_cast<float>(y);
|
||||||
new_img(u, v) += (*img)(x, y)
|
new_img(u, v) += (*img)(x, y)
|
||||||
* cos((2*fx + 1) / (2 * N))
|
* cos(((2*fx + 1) * fu * M_PI) / (2 * N))
|
||||||
* cos((2*fy + 1) / (2 * M));
|
* cos(((2*fy + 1) * fv * M_PI) / (2 * M));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (u == 0 && v == 0) {
|
if (u == 0 && v == 0) {
|
||||||
new_img(u, v) *= 0.5;
|
new_img(u, v) /= N;
|
||||||
} else if (u == 0 || v == 0) {
|
} else if (u == 0 || v == 0) {
|
||||||
new_img(u, v) /= sqrt(2);
|
new_img(u, v) *= sqrt(2) / N;
|
||||||
|
} else {
|
||||||
|
new_img(u, v) *= 2 / N;
|
||||||
}
|
}
|
||||||
new_img(u, v) *= 0.25;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
*img = new_img;
|
*img = new_img;
|
||||||
}
|
}
|
||||||
|
|
||||||
extern void DiscreteCosineXForm(Image<IPixel> *img, int blockSize) {
|
static void IDCT(Image<IPixel> *img) {
|
||||||
|
Image<IPixel> new_img = *img;
|
||||||
|
|
||||||
|
float N = static_cast<float>(img->GetWidth());
|
||||||
|
float M = static_cast<float>(img->GetHeight());
|
||||||
|
|
||||||
|
assert (N == M);
|
||||||
|
|
||||||
|
for (unsigned int y = 0; y < img->GetHeight(); ++y) {
|
||||||
|
for (unsigned int x = 0; x < img->GetWidth(); ++x) {
|
||||||
|
new_img(x, y) = 0.0f;
|
||||||
|
float fx = static_cast<float>(x);
|
||||||
|
float fy = static_cast<float>(y);
|
||||||
|
|
||||||
|
for (unsigned int v = 0; v < img->GetHeight(); ++v) {
|
||||||
|
for (unsigned int u = 0; u < img->GetWidth(); ++u) {
|
||||||
|
float fu = static_cast<float>(u);
|
||||||
|
float fv = static_cast<float>(v);
|
||||||
|
new_img(x, y) += (*img)(u, v)
|
||||||
|
* cos(((2*fx + 1) * fu * M_PI) / (2 * N))
|
||||||
|
* cos(((2*fy + 1) * fv * M_PI) / (2 * M));
|
||||||
|
|
||||||
|
if (u == 0 && v == 0) {
|
||||||
|
new_img(x, y) /= N;
|
||||||
|
} else if (u == 0 || v == 0) {
|
||||||
|
new_img(x, y) /= sqrt(2) / N;
|
||||||
|
} else {
|
||||||
|
new_img(x, y) *= 2 / N;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
*img = new_img;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void RunDCTBlockFn(Image<IPixel> *img, int blockSize, DCTBlockFn fn) {
|
||||||
|
assert (NULL != fn);
|
||||||
|
assert (0 < blockSize);
|
||||||
|
assert (static_cast<uint32>(blockSize) < img->GetWidth());
|
||||||
|
assert (static_cast<uint32>(blockSize) < img->GetHeight());
|
||||||
|
|
||||||
Image<IPixel> block(blockSize, blockSize);
|
Image<IPixel> block(blockSize, blockSize);
|
||||||
for (unsigned int j = 0; j < img->GetHeight(); j += blockSize) {
|
for (unsigned int j = 0; j < img->GetHeight(); j += blockSize) {
|
||||||
for (unsigned int i = 0; i < img->GetWidth(); i += blockSize) {
|
for (unsigned int i = 0; i < img->GetWidth(); i += blockSize) {
|
||||||
|
@ -660,8 +709,8 @@ extern void DiscreteCosineXForm(Image<IPixel> *img, int blockSize) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Transform it
|
// Run the function
|
||||||
DCT(&block);
|
fn(&block);
|
||||||
|
|
||||||
// Put it back in the original image
|
// Put it back in the original image
|
||||||
for (int y = 0; y < blockSize; ++y) {
|
for (int y = 0; y < blockSize; ++y) {
|
||||||
|
@ -684,5 +733,12 @@ extern void DiscreteCosineXForm(Image<IPixel> *img, int blockSize) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void DiscreteCosineXForm(Image<IPixel> *img, int blockSize) {
|
||||||
|
RunDCTBlockFn(img, blockSize, DCT);
|
||||||
|
}
|
||||||
|
|
||||||
|
void InvDiscreteCosineXForm(Image<IPixel> *img, int blockSize) {
|
||||||
|
RunDCTBlockFn(img, blockSize, IDCT);
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace FasTC
|
} // namespace FasTC
|
||||||
|
|
|
@ -211,3 +211,38 @@ TEST(Image, SplitImage) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST(Image, DCT) {
|
||||||
|
|
||||||
|
const uint32 w = 32;
|
||||||
|
const uint32 h = 32;
|
||||||
|
|
||||||
|
FasTC::Image<FasTC::IPixel> img(w, h);
|
||||||
|
for (uint32 j = 0; j < h; ++j) {
|
||||||
|
for (uint32 i = 0; i < w; ++i) {
|
||||||
|
img(i, j) = static_cast<FasTC::IPixel>(i + j);
|
||||||
|
// img(i, j) = static_cast<FasTC::IPixel>(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
FasTC::Image<FasTC::IPixel> orig(img);
|
||||||
|
|
||||||
|
// Make sure that taking the DCT and inverse DCT returns
|
||||||
|
// the same image...
|
||||||
|
FasTC::DiscreteCosineXForm(&img, 8);
|
||||||
|
|
||||||
|
// First make sure they're different
|
||||||
|
for (uint32 j = 0; j < h; ++j) {
|
||||||
|
for (uint32 i = 0; i < w; ++i) {
|
||||||
|
EXPECT_NE(img(i, j), orig(i, j));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
FasTC::InvDiscreteCosineXForm(&img, 8);
|
||||||
|
|
||||||
|
for (uint32 j = 0; j < h; ++j) {
|
||||||
|
for (uint32 i = 0; i < w; ++i) {
|
||||||
|
EXPECT_NEAR(img(i, j), orig(i, j), 1e-5);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue