Add support for input rescaling:

- round extents to power of two.
- clamp max extents.
Add output options pimpl.
Other misc API changes.
This commit is contained in:
castano 2007-12-17 03:51:34 +00:00
parent 817652c56c
commit cfa4913ae6
12 changed files with 872 additions and 191 deletions

View File

@ -15,13 +15,17 @@ SET(NVTT_SRCS
FastCompressDXT.cpp FastCompressDXT.cpp
QuickCompressDXT.h QuickCompressDXT.h
QuickCompressDXT.cpp QuickCompressDXT.cpp
SingleColorLookup.h
CompressionOptions.h CompressionOptions.h
CompressionOptions.cpp CompressionOptions.cpp
InputOptions.h InputOptions.h
InputOptions.cpp InputOptions.cpp
OutputOptions.h
OutputOptions.cpp OutputOptions.cpp
cuda/CudaUtils.h cuda/CudaUtils.h
cuda/CudaUtils.cpp cuda/CudaUtils.cpp
cuda/CudaMath.h
cuda/Bitmaps.h
cuda/CudaCompressDXT.h cuda/CudaCompressDXT.h
cuda/CudaCompressDXT.cpp) cuda/CudaCompressDXT.cpp)
@ -45,23 +49,62 @@ ENDIF(NVTT_SHARED)
TARGET_LINK_LIBRARIES(nvtt ${LIBS} nvcore nvmath nvimage squish) TARGET_LINK_LIBRARIES(nvtt ${LIBS} nvcore nvmath nvimage squish)
INSTALL(TARGETS nvtt
RUNTIME DESTINATION bin
LIBRARY DESTINATION lib
ARCHIVE DESTINATION lib/static)
INSTALL(FILES nvtt.h DESTINATION include/nvtt)
# test executables # test executables
ADD_EXECUTABLE(nvcompress tools/compress.cpp) ADD_EXECUTABLE(nvcompress tools/compress.cpp tools/cmdline.h)
TARGET_LINK_LIBRARIES(nvcompress nvcore nvmath nvimage nvtt) TARGET_LINK_LIBRARIES(nvcompress nvcore nvmath nvimage nvtt)
ADD_EXECUTABLE(nvdecompress tools/decompress.cpp) ADD_EXECUTABLE(nvdecompress tools/decompress.cpp tools/cmdline.h)
TARGET_LINK_LIBRARIES(nvdecompress nvcore nvmath nvimage) TARGET_LINK_LIBRARIES(nvdecompress nvcore nvmath nvimage)
ADD_EXECUTABLE(nvddsinfo tools/ddsinfo.cpp) ADD_EXECUTABLE(nvddsinfo tools/ddsinfo.cpp tools/cmdline.h)
TARGET_LINK_LIBRARIES(nvddsinfo nvcore nvmath nvimage) TARGET_LINK_LIBRARIES(nvddsinfo nvcore nvmath nvimage)
ADD_EXECUTABLE(nvimgdiff tools/imgdiff.cpp) ADD_EXECUTABLE(nvimgdiff tools/imgdiff.cpp tools/cmdline.h)
TARGET_LINK_LIBRARIES(nvimgdiff nvcore nvmath nvimage) TARGET_LINK_LIBRARIES(nvimgdiff nvcore nvmath nvimage)
ADD_EXECUTABLE(nvassemble tools/assemble.cpp) ADD_EXECUTABLE(nvassemble tools/assemble.cpp tools/cmdline.h)
TARGET_LINK_LIBRARIES(nvassemble nvcore nvmath nvimage) TARGET_LINK_LIBRARIES(nvassemble nvcore nvmath nvimage)
INSTALL(TARGETS nvcompress nvdecompress nvddsinfo nvimgdiff nvassemble DESTINATION bin) ADD_EXECUTABLE(filtertest tests/filtertest.cpp tools/cmdline.h)
TARGET_LINK_LIBRARIES(filtertest nvcore nvmath nvimage)
ADD_EXECUTABLE(nvzoom tools/resize.cpp tools/cmdline.h)
TARGET_LINK_LIBRARIES(nvzoom nvcore nvmath nvimage)
INSTALL(TARGETS nvcompress nvdecompress nvddsinfo nvimgdiff nvassemble nvzoom DESTINATION bin)
# UI tools
IF(QT4_FOUND)
SET(QT_USE_QTOPENGL TRUE)
INCLUDE_DIRECTORIES(${QT_INCLUDE_DIR} ${CMAKE_CURRENT_BINARY_DIR})
SET(SRCS
tools/main.cpp
tools/configdialog.h
tools/configdialog.cpp)
SET(LIBS
nvtt
${QT_QTCORE_LIBRARY}
${QT_QTGUI_LIBRARY}
${QT_QTOPENGL_LIBRARY})
QT4_WRAP_UI(UICS tools/configdialog.ui)
QT4_WRAP_CPP(MOCS tools/configdialog.h)
#QT4_ADD_RESOURCES(RCCS tools/configdialog.rc)
ADD_EXECUTABLE(nvcompressui MACOSX_BUNDLE ${SRCS} ${UICS} ${MOCS})
TARGET_LINK_LIBRARIES(nvcompressui ${LIBS})
ENDIF(QT4_FOUND)

View File

@ -32,6 +32,7 @@
#include "FastCompressDXT.h" #include "FastCompressDXT.h"
#include "QuickCompressDXT.h" #include "QuickCompressDXT.h"
#include "CompressionOptions.h" #include "CompressionOptions.h"
#include "OutputOptions.h"
// squish // squish
#include "squish/colourset.h" #include "squish/colourset.h"
@ -55,7 +56,7 @@ using namespace nv;
using namespace nvtt; using namespace nvtt;
void nv::fastCompressDXT1(const Image * image, const OutputOptions & outputOptions) void nv::fastCompressDXT1(const Image * image, const OutputOptions::Private & outputOptions)
{ {
const uint w = image->width(); const uint w = image->width();
const uint h = image->height(); const uint h = image->height();
@ -77,7 +78,7 @@ void nv::fastCompressDXT1(const Image * image, const OutputOptions & outputOptio
} }
void nv::fastCompressDXT1a(const Image * image, const OutputOptions & outputOptions) void nv::fastCompressDXT1a(const Image * image, const OutputOptions::Private & outputOptions)
{ {
const uint w = image->width(); const uint w = image->width();
const uint h = image->height(); const uint h = image->height();
@ -99,7 +100,7 @@ void nv::fastCompressDXT1a(const Image * image, const OutputOptions & outputOpti
} }
void nv::fastCompressDXT3(const Image * image, const nvtt::OutputOptions & outputOptions) void nv::fastCompressDXT3(const Image * image, const nvtt::OutputOptions::Private & outputOptions)
{ {
const uint w = image->width(); const uint w = image->width();
const uint h = image->height(); const uint h = image->height();
@ -120,7 +121,7 @@ void nv::fastCompressDXT3(const Image * image, const nvtt::OutputOptions & outpu
} }
void nv::fastCompressDXT5(const Image * image, const nvtt::OutputOptions & outputOptions) void nv::fastCompressDXT5(const Image * image, const nvtt::OutputOptions::Private & outputOptions)
{ {
const uint w = image->width(); const uint w = image->width();
const uint h = image->height(); const uint h = image->height();
@ -141,7 +142,7 @@ void nv::fastCompressDXT5(const Image * image, const nvtt::OutputOptions & outpu
} }
void nv::fastCompressDXT5n(const Image * image, const nvtt::OutputOptions & outputOptions) void nv::fastCompressDXT5n(const Image * image, const nvtt::OutputOptions::Private & outputOptions)
{ {
const uint w = image->width(); const uint w = image->width();
const uint h = image->height(); const uint h = image->height();
@ -166,14 +167,14 @@ void nv::fastCompressDXT5n(const Image * image, const nvtt::OutputOptions & outp
} }
void nv::fastCompressBC4(const Image * image, const nvtt::OutputOptions & outputOptions) void nv::fastCompressBC4(const Image * image, const nvtt::OutputOptions::Private & outputOptions)
{ {
// @@ TODO // @@ TODO
// compress red channel (X) // compress red channel (X)
} }
void nv::fastCompressBC5(const Image * image, const nvtt::OutputOptions & outputOptions) void nv::fastCompressBC5(const Image * image, const nvtt::OutputOptions::Private & outputOptions)
{ {
// @@ TODO // @@ TODO
// compress red, green channels (X,Y) // compress red, green channels (X,Y)
@ -192,7 +193,7 @@ void nv::doPrecomputation()
} }
void nv::compressDXT1(const Image * image, const OutputOptions & outputOptions, const CompressionOptions::Private & compressionOptions) void nv::compressDXT1(const Image * image, const OutputOptions::Private & outputOptions, const CompressionOptions::Private & compressionOptions)
{ {
const uint w = image->width(); const uint w = image->width();
const uint h = image->height(); const uint h = image->height();
@ -225,7 +226,7 @@ void nv::compressDXT1(const Image * image, const OutputOptions & outputOptions,
} }
void nv::compressDXT3(const Image * image, const OutputOptions & outputOptions, const CompressionOptions::Private & compressionOptions) void nv::compressDXT3(const Image * image, const OutputOptions::Private & outputOptions, const CompressionOptions::Private & compressionOptions)
{ {
const uint w = image->width(); const uint w = image->width();
const uint h = image->height(); const uint h = image->height();
@ -254,7 +255,7 @@ void nv::compressDXT3(const Image * image, const OutputOptions & outputOptions,
} }
} }
void nv::compressDXT5(const Image * image, const OutputOptions & outputOptions, const CompressionOptions::Private & compressionOptions) void nv::compressDXT5(const Image * image, const OutputOptions::Private & outputOptions, const CompressionOptions::Private & compressionOptions)
{ {
const uint w = image->width(); const uint w = image->width();
const uint h = image->height(); const uint h = image->height();
@ -292,7 +293,7 @@ void nv::compressDXT5(const Image * image, const OutputOptions & outputOptions,
} }
void nv::compressDXT5n(const Image * image, const OutputOptions & outputOptions, const CompressionOptions::Private & compressionOptions) void nv::compressDXT5n(const Image * image, const OutputOptions::Private & outputOptions, const CompressionOptions::Private & compressionOptions)
{ {
const uint w = image->width(); const uint w = image->width();
const uint h = image->height(); const uint h = image->height();
@ -328,7 +329,7 @@ void nv::compressDXT5n(const Image * image, const OutputOptions & outputOptions,
} }
void nv::compressBC4(const Image * image, const nvtt::OutputOptions & outputOptions, const CompressionOptions::Private & compressionOptions) void nv::compressBC4(const Image * image, const nvtt::OutputOptions::Private & outputOptions, const CompressionOptions::Private & compressionOptions)
{ {
const uint w = image->width(); const uint w = image->width();
const uint h = image->height(); const uint h = image->height();
@ -380,7 +381,7 @@ void nv::compressBC4(const Image * image, const nvtt::OutputOptions & outputOpti
} }
void nv::compressBC5(const Image * image, const nvtt::OutputOptions & outputOptions, const CompressionOptions::Private & compressionOptions) void nv::compressBC5(const Image * image, const nvtt::OutputOptions::Private & outputOptions, const CompressionOptions::Private & compressionOptions)
{ {
const uint w = image->width(); const uint w = image->width();
const uint h = image->height(); const uint h = image->height();
@ -423,7 +424,7 @@ void nv::compressBC5(const Image * image, const nvtt::OutputOptions & outputOpti
#if defined(HAVE_S3QUANT) #if defined(HAVE_S3QUANT)
void nv::s3CompressDXT1(const Image * image, const nvtt::OutputOptions & outputOptions) void nv::s3CompressDXT1(const Image * image, const nvtt::OutputOptions::Private & outputOptions)
{ {
const uint w = image->width(); const uint w = image->width();
const uint h = image->height(); const uint h = image->height();
@ -518,7 +519,7 @@ void nv::s3CompressDXT1(const Image * image, const nvtt::OutputOptions & outputO
#if defined(HAVE_ATITC) #if defined(HAVE_ATITC)
void nv::atiCompressDXT1(const Image * image, const OutputOptions & outputOptions) void nv::atiCompressDXT1(const Image * image, const OutputOptions::Private & outputOptions)
{ {
// Init source texture // Init source texture
ATI_TC_Texture srcTexture; ATI_TC_Texture srcTexture;

View File

@ -35,29 +35,29 @@ namespace nv
void doPrecomputation(); void doPrecomputation();
// Fast compressors. // Fast compressors.
void fastCompressDXT1(const Image * image, const nvtt::OutputOptions & outputOptions); void fastCompressDXT1(const Image * image, const nvtt::OutputOptions::Private & outputOptions);
void fastCompressDXT1a(const Image * image, const nvtt::OutputOptions & outputOptions); void fastCompressDXT1a(const Image * image, const nvtt::OutputOptions::Private & outputOptions);
void fastCompressDXT3(const Image * image, const nvtt::OutputOptions & outputOptions); void fastCompressDXT3(const Image * image, const nvtt::OutputOptions::Private & outputOptions);
void fastCompressDXT5(const Image * image, const nvtt::OutputOptions & outputOptions); void fastCompressDXT5(const Image * image, const nvtt::OutputOptions::Private & outputOptions);
void fastCompressDXT5n(const Image * image, const nvtt::OutputOptions & outputOptions); void fastCompressDXT5n(const Image * image, const nvtt::OutputOptions::Private & outputOptions);
void fastCompressBC4(const Image * image, const nvtt::OutputOptions & outputOptions); void fastCompressBC4(const Image * image, const nvtt::OutputOptions::Private & outputOptions);
void fastCompressBC5(const Image * image, const nvtt::OutputOptions & outputOptions); void fastCompressBC5(const Image * image, const nvtt::OutputOptions::Private & outputOptions);
// Normal compressors. // Normal compressors.
void compressDXT1(const Image * image, const nvtt::OutputOptions & outputOptions, const nvtt::CompressionOptions::Private & compressionOptions); void compressDXT1(const Image * image, const nvtt::OutputOptions::Private & outputOptions, const nvtt::CompressionOptions::Private & compressionOptions);
void compressDXT3(const Image * image, const nvtt::OutputOptions & outputOptions, const nvtt::CompressionOptions::Private & compressionOptions); void compressDXT3(const Image * image, const nvtt::OutputOptions::Private & outputOptions, const nvtt::CompressionOptions::Private & compressionOptions);
void compressDXT5(const Image * image, const nvtt::OutputOptions & outputOptions, const nvtt::CompressionOptions::Private & compressionOptions); void compressDXT5(const Image * image, const nvtt::OutputOptions::Private & outputOptions, const nvtt::CompressionOptions::Private & compressionOptions);
void compressDXT5n(const Image * image, const nvtt::OutputOptions & outputOptions, const nvtt::CompressionOptions::Private & compressionOptions); void compressDXT5n(const Image * image, const nvtt::OutputOptions::Private & outputOptions, const nvtt::CompressionOptions::Private & compressionOptions);
void compressBC4(const Image * image, const nvtt::OutputOptions & outputOptions, const nvtt::CompressionOptions::Private & compressionOptions); void compressBC4(const Image * image, const nvtt::OutputOptions::Private & outputOptions, const nvtt::CompressionOptions::Private & compressionOptions);
void compressBC5(const Image * image, const nvtt::OutputOptions & outputOptions, const nvtt::CompressionOptions::Private & compressionOptions); void compressBC5(const Image * image, const nvtt::OutputOptions::Private & outputOptions, const nvtt::CompressionOptions::Private & compressionOptions);
// External compressors. // External compressors.
#if defined(HAVE_S3QUANT) #if defined(HAVE_S3QUANT)
void s3CompressDXT1(const Image * image, const nvtt::OutputOptions & outputOptions); void s3CompressDXT1(const Image * image, const nvtt::OutputOptions::Private & outputOptions);
#endif #endif
#if defined(HAVE_ATITC) #if defined(HAVE_ATITC)
void atiCompressDXT1(const Image * image, const nvtt::OutputOptions & outputOptions); void atiCompressDXT1(const Image * image, const nvtt::OutputOptions::Private & outputOptions);
#endif #endif
} // nv namespace } // nv namespace

View File

@ -29,7 +29,7 @@
#include "CompressRGB.h" #include "CompressRGB.h"
#include "CompressionOptions.h" #include "CompressionOptions.h"
#include "OutputOptions.h"
using namespace nv; using namespace nv;
using namespace nvtt; using namespace nvtt;
@ -59,7 +59,7 @@ namespace
// Pixel format converter. // Pixel format converter.
void nv::compressRGB(const Image * image, const OutputOptions & outputOptions, const CompressionOptions::Private & compressionOptions) void nv::compressRGB(const Image * image, const OutputOptions::Private & outputOptions, const CompressionOptions::Private & compressionOptions)
{ {
nvCheck(image != NULL); nvCheck(image != NULL);

View File

@ -19,21 +19,21 @@
// HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, // HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
// WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING // WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
// OTHER DEALINGS IN THE SOFTWARE. // OTHER DEALINGS IN THE SOFTWARE.
#ifndef NV_TT_COMPRESSRGB_H #ifndef NV_TT_COMPRESSRGB_H
#define NV_TT_COMPRESSRGB_H #define NV_TT_COMPRESSRGB_H
#include "nvtt.h" #include "nvtt.h"
namespace nv namespace nv
{ {
class Image; class Image;
// Pixel format converter. // Pixel format converter.
void compressRGB(const Image * image, const nvtt::OutputOptions & outputOptions, const nvtt::CompressionOptions::Private & compressionOptions); void compressRGB(const Image * image, const nvtt::OutputOptions::Private & outputOptions, const nvtt::CompressionOptions::Private & compressionOptions);
} // nv namespace } // nv namespace
#endif // NV_TT_COMPRESSDXT_H #endif // NV_TT_COMPRESSDXT_H

View File

@ -87,12 +87,18 @@ void CompressionOptions::setColorWeights(float red, float green, float blue)
} }
/// Enable or disable hardware compression. /// Enable or disable CUDA compression.
void CompressionOptions::enableHardwareCompression(bool enable) void CompressionOptions::enableHardwareCompression(bool enable)
{ {
m.useCuda = enable; m.useCuda = enable;
} }
/// Enable or disable CUDA compression.
void CompressionOptions::enableCudaCompression(bool enable)
{
m.useCuda = enable;
}
/// Set color mask to describe the RGB/RGBA format. /// Set color mask to describe the RGB/RGBA format.
void CompressionOptions::setPixelFormat(uint bitcount, uint rmask, uint gmask, uint bmask, uint amask) void CompressionOptions::setPixelFormat(uint bitcount, uint rmask, uint gmask, uint bmask, uint amask)

View File

@ -34,9 +34,9 @@ using namespace nvtt;
namespace namespace
{ {
static int countMipmaps(int w, int h, int d) static uint countMipmaps(int w, int h, int d)
{ {
int mipmap = 0; uint mipmap = 0;
while (w != 1 || h != 1 || d != 1) { while (w != 1 || h != 1 || d != 1) {
w = max(1, w / 2); w = max(1, w / 2);
@ -48,6 +48,27 @@ namespace
return mipmap + 1; return mipmap + 1;
} }
// 1 -> 1, 2 -> 2, 3 -> 2, 4 -> 4, 5 -> 4, ...
static uint previousPowerOfTwo(const uint v)
{
return nextPowerOfTwo(v + 1) / 2;
}
static uint nearestPowerOfTwo(const uint v)
{
const uint np2 = nextPowerOfTwo(v);
const uint pp2 = previousPowerOfTwo(v);
if (np2 - v <= v - pp2)
{
return np2;
}
else
{
return pp2;
}
}
} // namespace } // namespace
@ -69,7 +90,7 @@ InputOptions::~InputOptions()
// Reset input options. // Reset input options.
void InputOptions::reset() void InputOptions::reset()
{ {
m.wrapMode = WrapMode_Repeat; m.wrapMode = WrapMode_Mirror;
m.textureType = TextureType_2D; m.textureType = TextureType_2D;
m.inputFormat = InputFormat_BGRA_8UB; m.inputFormat = InputFormat_BGRA_8UB;
@ -78,26 +99,30 @@ void InputOptions::reset()
m.binaryAlpha = false; m.binaryAlpha = false;
m.alphaThreshold = 127; m.alphaThreshold = 127;
m.alphaTransparency = true; m.alphaMode = AlphaMode_Transparency;
m.inputGamma = 2.2f; m.inputGamma = 2.2f;
m.outputGamma = 2.2f; m.outputGamma = 2.2f;
m.colorTransform = ColorTransform_None; m.colorTransform = ColorTransform_None;
m.linearTransform = Matrix(identity); m.linearTransform = Matrix(identity);
m.generateMipmaps = false; m.generateMipmaps = false;
m.maxLevel = -1; m.maxLevel = -1;
m.mipmapFilter = MipmapFilter_Box; m.mipmapFilter = MipmapFilter_Box;
m.kaiserWidth = 10; m.kaiserWidth = 3;
m.kaiserAlpha = 8.0f; m.kaiserAlpha = 4.0f;
m.kaiserStretch = 0.75f; m.kaiserStretch = 1.0f;
m.normalizeMipmaps = false; m.isNormalMap = false;
m.normalizeMipmaps = true;
m.convertToNormalMap = false; m.convertToNormalMap = false;
m.heightFactors.set(0.0f, 0.0f, 0.0f, 1.0f); m.heightFactors.set(0.0f, 0.0f, 0.0f, 1.0f);
m.bumpFrequencyScale = Vector4(1.0f, 0.5f, 0.25f, 0.125f) / (1.0f + 0.5f + 0.25f + 0.125f); m.bumpFrequencyScale = Vector4(1.0f, 0.5f, 0.25f, 0.125f) / (1.0f + 0.5f + 0.25f + 0.125f);
m.maxExtent = 0;
m.roundMode = RoundMode_None;
} }
@ -126,13 +151,13 @@ void InputOptions::setTextureLayout(TextureType type, int width, int height, int
m.images = new Private::Image[m.imageCount]; m.images = new Private::Image[m.imageCount];
for(int f = 0; f < m.faceCount; f++) for(uint f = 0; f < m.faceCount; f++)
{ {
int w = width; uint w = width;
int h = height; uint h = height;
int d = depth; uint d = depth;
for (int mipLevel = 0; mipLevel < m.mipmapCount; mipLevel++) for (uint mipLevel = 0; mipLevel < m.mipmapCount; mipLevel++)
{ {
Private::Image & img = m.images[f * m.mipmapCount + mipLevel]; Private::Image & img = m.images[f * m.mipmapCount + mipLevel];
img.width = w; img.width = w;
@ -143,9 +168,9 @@ void InputOptions::setTextureLayout(TextureType type, int width, int height, int
img.data = NULL; img.data = NULL;
w = max(1, w / 2); w = max(1U, w / 2);
h = max(1, h / 2); h = max(1U, h / 2);
d = max(1, d / 2); d = max(1U, d / 2);
} }
} }
} }
@ -188,10 +213,21 @@ bool InputOptions::setMipmapData(const void * data, int width, int height, int d
/// Describe the format of the input. /// Describe the format of the input.
void InputOptions::setFormat(InputFormat format, bool alphaTransparency) void InputOptions::setFormat(InputFormat format, /*deprecated*/bool alphaTransparency)
{ {
m.inputFormat = format; m.inputFormat = format;
m.alphaTransparency = alphaTransparency; //m.alphaTransparency = alphaTransparency;
}
void InputOptions::setFormat(InputFormat format)
{
m.inputFormat = format;
}
/// Set the way the input alpha channel is interpreted.
void InputOptions::setAlphaMode(AlphaMode alphaMode)
{
m.alphaMode = alphaMode;
} }
@ -219,7 +255,7 @@ void InputOptions::setMipmapping(bool generateMipmaps, MipmapFilter filter/*= Mi
} }
/// Set Kaiser filter parameters. /// Set Kaiser filter parameters.
void InputOptions::setKaiserParameters(int width, float alpha, float stretch) void InputOptions::setKaiserParameters(float width, float alpha, float stretch)
{ {
m.kaiserWidth = width; m.kaiserWidth = width;
m.kaiserAlpha = alpha; m.kaiserAlpha = alpha;
@ -233,6 +269,7 @@ void InputOptions::setKaiserParameters(int width, float alpha, float stretch)
/// the compressor. /// the compressor.
void InputOptions::setQuantization(bool colorDithering, bool alphaDithering, bool binaryAlpha, int alphaThreshold/*= 127*/) void InputOptions::setQuantization(bool colorDithering, bool alphaDithering, bool binaryAlpha, int alphaThreshold/*= 127*/)
{ {
nvCheck(alphaThreshold >= 0 && alphaThreshold < 256);
m.enableColorDithering = colorDithering; m.enableColorDithering = colorDithering;
m.enableAlphaDithering = alphaDithering; m.enableAlphaDithering = alphaDithering;
m.binaryAlpha = binaryAlpha; m.binaryAlpha = binaryAlpha;
@ -243,7 +280,7 @@ void InputOptions::setQuantization(bool colorDithering, bool alphaDithering, boo
/// Indicate whether input is a normal map or not. /// Indicate whether input is a normal map or not.
void InputOptions::setNormalMap(bool b) void InputOptions::setNormalMap(bool b)
{ {
m.normalMap = b; m.isNormalMap = b;
} }
/// Enable normal map conversion. /// Enable normal map conversion.
@ -287,3 +324,103 @@ void InputOptions::setLinearTransfrom(int channel, float w0, float w1, float w2,
Vector4 w(w0, w1, w2, w3); Vector4 w(w0, w1, w2, w3);
//m.linearTransform.setRow(channel, w); //m.linearTransform.setRow(channel, w);
} }
void InputOptions::setMaxExtents(int e)
{
nvDebugCheck(e > 0);
m.maxExtent = e;
}
void InputOptions::setRoundMode(RoundMode mode)
{
m.roundMode = mode;
}
void InputOptions::Private::computeTargetExtents() const
{
nvCheck(images != NULL);
uint maxExtent = this->maxExtent;
if (roundMode != RoundMode_None)
{
// rounded max extent should never be higher than original max extent.
maxExtent = previousPowerOfTwo(maxExtent);
}
uint w = images->width;
uint h = images->height;
uint d = images->depth;
nvDebugCheck(w > 0);
nvDebugCheck(h > 0);
nvDebugCheck(d > 0);
// Scale extents without changing aspect ratio.
uint maxwhd = max(max(w, h), d);
if (maxExtent != 0 && maxwhd > maxExtent)
{
w = max((w * maxExtent) / maxwhd, 1U);
h = max((h * maxExtent) / maxwhd, 1U);
d = max((d * maxExtent) / maxwhd, 1U);
}
// Round to power of two.
if (roundMode == RoundMode_ToNextPowerOfTwo)
{
w = nextPowerOfTwo(w);
h = nextPowerOfTwo(h);
d = nextPowerOfTwo(d);
}
else if (roundMode == RoundMode_ToNearestPowerOfTwo)
{
w = nearestPowerOfTwo(w);
h = nearestPowerOfTwo(h);
d = nearestPowerOfTwo(d);
}
else if (roundMode == RoundMode_ToPreviousPowerOfTwo)
{
w = previousPowerOfTwo(w);
h = previousPowerOfTwo(h);
d = previousPowerOfTwo(d);
}
this->targetWidth = w;
this->targetHeight = h;
this->targetDepth = d;
this->targetMipmapCount = countMipmaps(w, h, d);
}
// Return real number of mipmaps, including first level.
// computeTargetExtents should have been called before.
int InputOptions::Private::realMipmapCount() const
{
int mipmapCount = targetMipmapCount;
if (!generateMipmaps) mipmapCount = 1;
else if (maxLevel != -1 && maxLevel < mipmapCount - 1) mipmapCount = maxLevel + 1;
return mipmapCount;
}
// Called everytime max extents or rounding mode changes.
int InputOptions::Private::firstMipmap(int face) const
{
nvCheck(images != NULL);
// @@ Find the last image that's not NULL and is greater than target extents.
uint first = 0;
for (uint f = 0; f < mipmapCount; f++)
{
}
return first;
}

View File

@ -36,54 +36,65 @@ namespace nvtt
struct InputOptions::Private struct InputOptions::Private
{ {
Private() : images(NULL) {} Private() : images(NULL) {}
WrapMode wrapMode; WrapMode wrapMode;
TextureType textureType; TextureType textureType;
InputFormat inputFormat; InputFormat inputFormat;
AlphaMode alphaMode;
int faceCount; uint faceCount;
int mipmapCount; uint mipmapCount;
int imageCount; uint imageCount;
struct Image; struct Image;
Image * images; Image * images;
// Quantization. // Quantization.
bool enableColorDithering; bool enableColorDithering;
bool enableAlphaDithering; bool enableAlphaDithering;
bool binaryAlpha; bool binaryAlpha;
int alphaThreshold; // reference value used for binary alpha quantization. int alphaThreshold; // reference value used for binary alpha quantization.
bool alphaTransparency; // set to true if alpha is used for transparency.
// Gamma conversion. // Gamma conversion.
float inputGamma; float inputGamma;
float outputGamma; float outputGamma;
// Color transform. // Color transform.
ColorTransform colorTransform; ColorTransform colorTransform;
nv::Matrix linearTransform; nv::Matrix linearTransform;
// Mipmap generation options. // Mipmap generation options.
bool generateMipmaps; bool generateMipmaps;
int maxLevel; int maxLevel;
MipmapFilter mipmapFilter; MipmapFilter mipmapFilter;
// Kaiser filter parameters. // Kaiser filter parameters.
uint kaiserWidth; float kaiserWidth;
float kaiserAlpha; float kaiserAlpha;
float kaiserStretch; float kaiserStretch;
// Normal map options. // Normal map options.
bool normalMap; bool isNormalMap;
bool normalizeMipmaps; bool normalizeMipmaps;
bool convertToNormalMap; bool convertToNormalMap;
nv::Vector4 heightFactors; // Used for cone mapping too. nv::Vector4 heightFactors;
nv::Vector4 bumpFrequencyScale; nv::Vector4 bumpFrequencyScale;
// Cone map options. // Adjust extents.
bool convertToConeMap; uint maxExtent;
RoundMode roundMode;
// @@ These are computed in nvtt::compress, so they should be mutable or stored elsewhere...
mutable uint targetWidth;
mutable uint targetHeight;
mutable uint targetDepth;
mutable uint targetMipmapCount;
void computeTargetExtents() const;
int realMipmapCount() const;
int firstMipmap(int face) const;
}; };
// Internal image structure. // Internal image structure.

View File

@ -21,12 +21,105 @@
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
// OTHER DEALINGS IN THE SOFTWARE. // OTHER DEALINGS IN THE SOFTWARE.
#include "nvtt.h" #include "OutputOptions.h"
using namespace nvtt; using namespace nvtt;
OutputOptions::OutputOptions() : m(*new OutputOptions::Private())
{
reset();
}
OutputOptions::OutputOptions(OutputHandler * oh, ErrorHandler * eh) : m(*new OutputOptions::Private())
{
reset();
outputHandler = oh;
errorHandler = eh;
}
OutputOptions::~OutputOptions()
{
delete &m;
}
/// Set default output options. /// Set default output options.
void OutputOptions::reset() void OutputOptions::reset()
{ {
// endiannes = native... m.fileName.reset();
m.outputHandler = NULL;
m.errorHandler = NULL;
m.outputHeader = true;
outputHandler = NULL;
errorHandler = NULL;
outputHeader = true;
} }
/// Set output file name.
void OutputOptions::setFileName(const char * fileName)
{
m.fileName = fileName;
outputHandler = NULL;
}
/// Set output handler.
void OutputOptions::setOutputHandler(OutputHandler * outputHandler)
{
m.fileName.reset();
this->outputHandler = outputHandler;
}
/// Set error handler.
void OutputOptions::setErrorHandler(ErrorHandler * errorHandler)
{
this->errorHandler = errorHandler;
}
/// Set output header.
void OutputOptions::setOutputHeader(bool outputHeader)
{
this->outputHeader = outputHeader;
}
bool OutputOptions::Private::openFile() const
{
if (!fileName.isNull())
{
nvCheck(outputHandler == NULL);
DefaultOutputHandler * oh = new DefaultOutputHandler(fileName.str());
if (oh->stream.isError())
{
return false;
}
outputHandler = oh;
}
return true;
}
void OutputOptions::Private::closeFile() const
{
if (!fileName.isNull())
{
delete outputHandler;
outputHandler = NULL;
}
}
void nvtt::initOptions(OutputOptions * outputOptions)
{
nvDebugCheck(outputOptions != NULL);
OutputOptions::Private & pimpl = outputOptions->m;
pimpl.outputHandler = outputOptions->outputHandler;
pimpl.errorHandler = outputOptions->errorHandler;
pimpl.outputHeader = outputOptions->outputHeader;
}

76
src/nvtt/OutputOptions.h Normal file
View File

@ -0,0 +1,76 @@
// Copyright NVIDIA Corporation 2007 -- Ignacio Castano <icastano@nvidia.com>
//
// Permission is hereby granted, free of charge, to any person
// obtaining a copy of this software and associated documentation
// files (the "Software"), to deal in the Software without
// restriction, including without limitation the rights to use,
// copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the
// Software is furnished to do so, subject to the following
// conditions:
//
// The above copyright notice and this permission notice shall be
// included in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
// OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
// HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
// WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
// OTHER DEALINGS IN THE SOFTWARE.
#ifndef NV_TT_OUTPUTOPTIONS_H
#define NV_TT_OUTPUTOPTIONS_H
#include <nvcore/StrLib.h>
#include <nvcore/StdStream.h>
#include "nvtt.h"
namespace nvtt
{
struct DefaultOutputHandler : public nvtt::OutputHandler
{
DefaultOutputHandler(const char * fileName) : stream(fileName) {}
virtual ~DefaultOutputHandler()
{
}
virtual void mipmap(int size, int width, int height, int depth, int face, int miplevel)
{
// ignore.
}
// Output data.
virtual void writeData(const void * data, int size)
{
stream.serialize(const_cast<void *>(data), size);
}
nv::StdOutputStream stream;
};
struct OutputOptions::Private
{
nv::Path fileName;
mutable OutputHandler * outputHandler;
mutable ErrorHandler * errorHandler;
mutable bool outputHeader;
bool openFile() const;
void closeFile() const;
};
// @@ temporary hack to copy public attributes to pimpl.
void initOptions(OutputOptions * outputOptions);
} // nvtt namespace
#endif // NV_TT_OUTPUTOPTIONS_H

View File

@ -38,6 +38,7 @@
#include "CompressRGB.h" #include "CompressRGB.h"
#include "InputOptions.h" #include "InputOptions.h"
#include "CompressionOptions.h" #include "CompressionOptions.h"
#include "OutputOptions.h"
#include "cuda/CudaUtils.h" #include "cuda/CudaUtils.h"
#include "cuda/CudaCompressDXT.h" #include "cuda/CudaCompressDXT.h"
@ -96,24 +97,21 @@ namespace
// compress // compress
// //
static void outputHeader(const InputOptions::Private & inputOptions, const OutputOptions & outputOptions, const CompressionOptions::Private & compressionOptions) static void outputHeader(const InputOptions::Private & inputOptions, const OutputOptions::Private & outputOptions, const CompressionOptions::Private & compressionOptions)
{ {
// Output DDS header. // Output DDS header.
if (outputOptions.outputHandler != NULL && outputOptions.outputHeader) if (outputOptions.outputHandler != NULL && outputOptions.outputHeader)
{ {
DDSHeader header; DDSHeader header;
InputOptions::Private::Image * img = inputOptions.images; header.setWidth(inputOptions.targetWidth);
nvCheck(img != NULL); header.setHeight(inputOptions.targetHeight);
header.setWidth(img->width); int mipmapCount = inputOptions.realMipmapCount();
header.setHeight(img->height); nvDebugCheck(mipmapCount > 0);
header.setMipmapCount(mipmapCount - 1);
int mipmapCount = inputOptions.mipmapCount;
if (!inputOptions.generateMipmaps) mipmapCount = 0;
else if (inputOptions.maxLevel != -1 && inputOptions.maxLevel < mipmapCount) mipmapCount = inputOptions.maxLevel;
header.setMipmapCount(mipmapCount);
if (inputOptions.textureType == TextureType_2D) { if (inputOptions.textureType == TextureType_2D) {
header.setTexture2D(); header.setTexture2D();
} }
@ -122,17 +120,17 @@ static void outputHeader(const InputOptions::Private & inputOptions, const Outpu
} }
/*else if (inputOptions.textureType == TextureType_3D) { /*else if (inputOptions.textureType == TextureType_3D) {
header.setTexture3D(); header.setTexture3D();
header.setDepth(img->depth); header.setDepth(inputOptions.targetDepth);
}*/ }*/
if (compressionOptions.format == Format_RGBA) if (compressionOptions.format == Format_RGBA)
{ {
header.setPitch(4 * img->width); header.setPitch(4 * inputOptions.targetWidth);
header.setPixelFormat(compressionOptions.bitcount, compressionOptions.rmask, compressionOptions.gmask, compressionOptions.bmask, compressionOptions.amask); header.setPixelFormat(compressionOptions.bitcount, compressionOptions.rmask, compressionOptions.gmask, compressionOptions.bmask, compressionOptions.amask);
} }
else else
{ {
header.setLinearSize(computeImageSize(img->width, img->height, compressionOptions.bitcount, compressionOptions.format)); header.setLinearSize(computeImageSize(inputOptions.targetWidth, inputOptions.targetHeight, compressionOptions.bitcount, compressionOptions.format));
if (compressionOptions.format == Format_DXT1 || compressionOptions.format == Format_DXT1a) { if (compressionOptions.format == Format_DXT1 || compressionOptions.format == Format_DXT1a) {
header.setFourCC('D', 'X', 'T', '1'); header.setFourCC('D', 'X', 'T', '1');
@ -145,14 +143,14 @@ static void outputHeader(const InputOptions::Private & inputOptions, const Outpu
} }
else if (compressionOptions.format == Format_DXT5n) { else if (compressionOptions.format == Format_DXT5n) {
header.setFourCC('D', 'X', 'T', '5'); header.setFourCC('D', 'X', 'T', '5');
header.setNormalFlag(true); if (inputOptions.isNormalMap) header.setNormalFlag(true);
} }
else if (compressionOptions.format == Format_BC4) { else if (compressionOptions.format == Format_BC4) {
header.setFourCC('A', 'T', 'I', '1'); header.setFourCC('A', 'T', 'I', '1');
} }
else if (compressionOptions.format == Format_BC5) { else if (compressionOptions.format == Format_BC5) {
header.setFourCC('A', 'T', 'I', '2'); header.setFourCC('A', 'T', 'I', '2');
header.setNormalFlag(true); if (inputOptions.isNormalMap) header.setNormalFlag(true);
} }
} }
@ -175,7 +173,7 @@ static void outputHeader(const InputOptions::Private & inputOptions, const Outpu
} }
static bool compressMipmap(const Image * image, const OutputOptions & outputOptions, const CompressionOptions::Private & compressionOptions) static bool compressMipmap(const Image * image, const OutputOptions::Private & outputOptions, const CompressionOptions::Private & compressionOptions)
{ {
nvDebugCheck(image != NULL); nvDebugCheck(image != NULL);
@ -196,7 +194,6 @@ static bool compressMipmap(const Image * image, const OutputOptions & outputOpti
#if defined(HAVE_ATITC) #if defined(HAVE_ATITC)
if (compressionOptions.externalCompressor == "ati") if (compressionOptions.externalCompressor == "ati")
{ {
printf("ATI\n");
atiCompressDXT1(image, outputOptions); atiCompressDXT1(image, outputOptions);
} }
else else
@ -289,7 +286,7 @@ static FloatImage * toFloatImage(const Image * image, const InputOptions::Privat
FloatImage * floatImage = new FloatImage(image); FloatImage * floatImage = new FloatImage(image);
if (inputOptions.normalMap) if (inputOptions.isNormalMap)
{ {
// Expand normals. to [-1, 1] range. // Expand normals. to [-1, 1] range.
// floatImage->expandNormals(0); // floatImage->expandNormals(0);
@ -309,7 +306,14 @@ static Image * toFixedImage(const FloatImage * floatImage, const InputOptions::P
{ {
nvDebugCheck(floatImage != NULL); nvDebugCheck(floatImage != NULL);
return floatImage->createImageGammaCorrect(inputOptions.outputGamma); if (inputOptions.isNormalMap || inputOptions.outputGamma == 1.0f)
{
return floatImage->createImage();
}
else
{
return floatImage->createImageGammaCorrect(inputOptions.outputGamma);
}
} }
@ -331,13 +335,13 @@ static FloatImage * createMipmap(const FloatImage * floatImage, const InputOptio
else /*if (inputOptions.mipmapFilter == MipmapFilter_Kaiser)*/ else /*if (inputOptions.mipmapFilter == MipmapFilter_Kaiser)*/
{ {
nvDebugCheck(inputOptions.mipmapFilter == MipmapFilter_Kaiser); nvDebugCheck(inputOptions.mipmapFilter == MipmapFilter_Kaiser);
KaiserFilter filter(float(inputOptions.kaiserWidth)); KaiserFilter filter(inputOptions.kaiserWidth);
filter.setParameters(inputOptions.kaiserAlpha, inputOptions.kaiserStretch); filter.setParameters(inputOptions.kaiserAlpha, inputOptions.kaiserStretch);
result = floatImage->downSample(filter, (FloatImage::WrapMode)inputOptions.wrapMode); result = floatImage->downSample(filter, (FloatImage::WrapMode)inputOptions.wrapMode);
} }
// Normalize mipmap. // Normalize mipmap.
if (inputOptions.normalizeMipmaps) if ((inputOptions.isNormalMap || inputOptions.convertToNormalMap) && inputOptions.normalizeMipmaps)
{ {
normalizeNormalMap(result); normalizeNormalMap(result);
} }
@ -383,50 +387,297 @@ static void quantize(Image * img, const InputOptions::Private & inputOptions, Fo
} }
} }
// Process the input, convert to normal map, normalize or convert to linear space.
static FloatImage * processInput(const InputOptions::Private & inputOptions, int idx)
{
const InputOptions::Private::Image & mipmap = inputOptions.images[idx];
if (inputOptions.convertToNormalMap)
{
// Scale height factor by 1 / 2 ^ m // @@ Compute scale factor exactly...
Vector4 heightScale = inputOptions.heightFactors / float(1 << idx);
return createNormalMap(mipmap.data.ptr(), (FloatImage::WrapMode)inputOptions.wrapMode, heightScale, inputOptions.bumpFrequencyScale);
}
else if (inputOptions.isNormalMap)
{
if (inputOptions.normalizeMipmaps)
{
FloatImage * img = new FloatImage(mipmap.data.ptr());
img->normalize(0);
return img;
}
}
else
{
if (inputOptions.inputGamma != inputOptions.outputGamma)
{
FloatImage * img = new FloatImage(mipmap.data.ptr());
img->toLinear(0, 3, inputOptions.inputGamma);
return img;
}
}
return NULL;
}
/// Compress the input texture with the given compression options.
bool nvtt::compress(const InputOptions & inputOptions, const OutputOptions & outputOptions, const CompressionOptions & compressionOptions)
struct ImagePair
{
ImagePair() : m_floatImage(NULL), m_fixedImage(NULL), m_deleteFixedImage(false) {}
~ImagePair()
{
if (m_deleteFixedImage) {
delete m_fixedImage;
}
}
void setFloatImage(FloatImage * image)
{
m_floatImage = image;
if (m_deleteFixedImage) delete m_fixedImage;
m_fixedImage = NULL;
}
void setFixedImage(Image * image, bool deleteImage)
{
m_floatImage = NULL;
if (m_deleteFixedImage) delete m_fixedImage;
m_fixedImage = image;
m_deleteFixedImage = deleteImage;
}
FloatImage * floatImage() const { return m_floatImage.ptr(); }
Image * fixedImage() const { return m_fixedImage; }
void toFixed(const InputOptions::Private & inputOptions)
{
if (m_floatImage != NULL)
{
// Convert to fixed.
m_fixedImage = toFixedImage(m_floatImage.ptr(), inputOptions);
}
}
private:
AutoPtr<FloatImage> m_floatImage;
Image * m_fixedImage;
bool m_deleteFixedImage;
};
// Find the first mipmap provided that is greater or equal to the target image size.
static int findMipmap(const InputOptions::Private & inputOptions, uint f, int firstMipmap, uint w, uint h, uint d)
{
int bestIdx = -1;
for (int m = firstMipmap; m < inputOptions.mipmapCount; m++)
{
int idx = f * inputOptions.mipmapCount + m;
const InputOptions::Private::Image & mipmap = inputOptions.images[idx];
if (mipmap.width >= w && mipmap.height >= h && mipmap.depth >= d)
{
if (mipmap.data != NULL)
{
bestIdx = idx;
}
}
else
{
// Do not look further down.
break;
}
}
return bestIdx;
}
static int findImage(const InputOptions::Private & inputOptions, uint f, uint w, uint h, uint d, int inputImageIdx, ImagePair * pair)
{
nvDebugCheck(w > 0 && h > 0);
nvDebugCheck(inputImageIdx >= 0 && inputImageIdx < inputOptions.mipmapCount);
nvDebugCheck(pair != NULL);
int bestIdx = findMipmap(inputOptions, f, inputImageIdx, w, h, d);
const InputOptions::Private::Image & mipmap = inputOptions.images[bestIdx];
if (mipmap.width == w && mipmap.height == h && mipmap.depth == d)
{
// Generate from input image.
AutoPtr<FloatImage> processedImage( processInput(inputOptions, bestIdx) );
if (processedImage != NULL)
{
pair->setFloatImage(processedImage.release());
}
else
{
pair->setFixedImage(mipmap.data.ptr(), false);
}
return bestIdx;
}
else
{
if (pair->floatImage() == NULL && pair->fixedImage() == NULL)
{
// Generate from input image and resize.
AutoPtr<FloatImage> processedImage( processInput(inputOptions, bestIdx) );
if (processedImage == NULL)
{
processedImage = new FloatImage(mipmap.data.ptr());
}
// Resize image. @@ Add more filters. @@ Distinguish between downscaling and reconstruction filters.
BoxFilter boxFilter;
pair->setFloatImage(processedImage->downSample(boxFilter, w, h, (FloatImage::WrapMode)inputOptions.wrapMode));
}
else
{
// Generate from previous mipmap.
if (pair->floatImage() == NULL)
{
nvDebugCheck(pair->fixedImage() != NULL);
pair->setFloatImage(toFloatImage(pair->fixedImage(), inputOptions));
}
// Create mipmap.
pair->setFloatImage(createMipmap(pair->floatImage(), inputOptions));
}
}
}
static bool compressMipmaps(uint f, const InputOptions::Private & inputOptions, const OutputOptions::Private & outputOptions, const CompressionOptions::Private & compressionOptions)
{
uint w = inputOptions.targetWidth;
uint h = inputOptions.targetHeight;
uint d = inputOptions.targetDepth;
int inputImageIdx = findMipmap(inputOptions, f, 0, w, h, d);
if (inputImageIdx == -1)
{
// First mipmap missing.
if (outputOptions.errorHandler != NULL) outputOptions.errorHandler->error(Error_InvalidInput);
return false;
}
ImagePair pair;
for (uint m = 0; m < inputOptions.mipmapCount; m++)
{
if (outputOptions.outputHandler)
{
int size = computeImageSize(w, h, compressionOptions.bitcount, compressionOptions.format);
outputOptions.outputHandler->mipmap(size, w, h, d, f, m);
}
inputImageIdx = findImage(inputOptions, f, w, h, d, inputImageIdx, &pair);
// @@ Where to do the color transform?
// - Color transform may not be linear, so we cannot do before computing mipmaps.
// - Should be done in linear space, that is, after gamma correction.
pair.toFixed(inputOptions);
// @@ Quantization should be done in compressMipmap! @@ It should not modify the input image!!!
quantize(pair.fixedImage(), inputOptions, compressionOptions.format);
compressMipmap(pair.fixedImage(), outputOptions, compressionOptions);
// Compute extents of next mipmap:
w = max(1U, w / 2);
h = max(1U, h / 2);
d = max(1U, d / 2);
}
return true;
}
static bool compress(const InputOptions::Private & inputOptions, const OutputOptions::Private & outputOptions, const CompressionOptions::Private & compressionOptions)
{ {
// Make sure enums match. // Make sure enums match.
nvStaticCheck(FloatImage::WrapMode_Clamp == (FloatImage::WrapMode)WrapMode_Clamp); nvStaticCheck(FloatImage::WrapMode_Clamp == (FloatImage::WrapMode)WrapMode_Clamp);
nvStaticCheck(FloatImage::WrapMode_Mirror == (FloatImage::WrapMode)WrapMode_Mirror); nvStaticCheck(FloatImage::WrapMode_Mirror == (FloatImage::WrapMode)WrapMode_Mirror);
nvStaticCheck(FloatImage::WrapMode_Repeat == (FloatImage::WrapMode)WrapMode_Repeat); nvStaticCheck(FloatImage::WrapMode_Repeat == (FloatImage::WrapMode)WrapMode_Repeat);
// Output DDS header. // Get output handler.
outputHeader(inputOptions.m, outputOptions, compressionOptions.m); if (!outputOptions.openFile())
Format format = compressionOptions.m.format;
const uint bitCount = compressionOptions.m.bitcount;
for (int f = 0; f < inputOptions.m.faceCount; f++)
{ {
if (outputOptions.errorHandler) outputOptions.errorHandler->error(Error_FileOpen);
// @@ Should return here?
}
inputOptions.computeTargetExtents();
uint mipmapCount = inputOptions.realMipmapCount();
nvDebugCheck(mipmapCount > 0);
// Output DDS header.
outputHeader(inputOptions, outputOptions, compressionOptions);
for (uint f = 0; f < inputOptions.faceCount; f++)
{
if (!compressMipmaps(f, inputOptions, outputOptions, compressionOptions))
{
return false;
}
/*
Image * lastImage = NULL; Image * lastImage = NULL;
AutoPtr<FloatImage> floatImage(NULL); AutoPtr<FloatImage> floatImage(NULL);
for (int m = 0; m < inputOptions.m.mipmapCount; m++) uint w = inputOptions.targetWidth;
uint h = inputOptions.targetHeight;
uint d = inputOptions.targetDepth;
for (uint m = 0; m < mipmapCount; m++)
{ {
int idx = f * inputOptions.m.mipmapCount + m;
InputOptions::Private::Image & mipmap = inputOptions.m.images[idx];
if (outputOptions.outputHandler) if (outputOptions.outputHandler)
{ {
int size = computeImageSize(mipmap.width, mipmap.height, bitCount, format); int size = computeImageSize(w, h, bitCount, format);
outputOptions.outputHandler->mipmap(size, mipmap.width, mipmap.height, mipmap.depth, mipmap.face, mipmap.mipLevel); outputOptions.outputHandler->mipmap(size, w, h, d, f, m);
} }
Image * img; // Image to compress. // @@ Write a more sofisticated get input image, that:
// - looks for the nearest image in the input mipmap chain, resizes it to desired extents.
// - uses previous floating point image, if available.
// - uses previous byte image if available.
int idx = f * inputOptions.mipmapCount + m;
InputOptions::Private::Image & mipmap = inputOptions.images[idx];
// @@ Prescale not implemented yet.
nvCheck(w == mipmap.width);
nvCheck(h == mipmap.height);
nvCheck(d == mipmap.depth);
Image * img = NULL; // Image to compress.
if (mipmap.data != NULL) // Mipmap provided. if (mipmap.data != NULL) // Mipmap provided.
{ {
// Convert to normal map. // Convert to normal map.
if (inputOptions.m.convertToNormalMap) if (inputOptions.convertToNormalMap)
{ {
floatImage = createNormalMap(mipmap.data.ptr(), (FloatImage::WrapMode)inputOptions.m.wrapMode, inputOptions.m.heightFactors, inputOptions.m.bumpFrequencyScale); // Scale height factor by 1 / 2 ^ m
Vector4 heightScale = inputOptions.heightFactors / float(1 << m);
floatImage = createNormalMap(mipmap.data.ptr(), (FloatImage::WrapMode)inputOptions.wrapMode, heightScale, inputOptions.bumpFrequencyScale);
} }
/*else if (inputOptions.m.convertToConeMap)
{
floatImage = createConeMap(mipmap.data, inputOptions.m.heightFactors);
}*/
else else
{ {
lastImage = img = mipmap.data.ptr(); lastImage = img = mipmap.data.ptr();
@ -446,70 +697,88 @@ bool nvtt::compress(const InputOptions & inputOptions, const OutputOptions & out
if (floatImage == NULL) if (floatImage == NULL)
{ {
nvDebugCheck(lastImage != NULL); nvDebugCheck(lastImage != NULL);
floatImage = toFloatImage(lastImage, inputOptions.m); floatImage = toFloatImage(lastImage, inputOptions);
} }
// Create mipmap. // Create mipmap.
floatImage = createMipmap(floatImage.ptr(), inputOptions.m); floatImage = createMipmap(floatImage.ptr(), inputOptions);
} }
if (floatImage != NULL) if (floatImage != NULL)
{ {
// Convert to fixed. // Convert to fixed.
img = toFixedImage(floatImage.ptr(), inputOptions.m); img = toFixedImage(floatImage.ptr(), inputOptions);
} }
// @@ Where to do the color transform? // @@ Where to do the color transform?
// - Color transform may not be linear, so we cannot do before computing mipmaps. // - Color transform may not be linear, so we cannot do before computing mipmaps.
// - Should be done in linear space, that is, after gamma correction. // - Should be done in linear space, that is, after gamma correction.
// @@ Error! gamma correction is not performed when mipmap data provied. // @@ Error! gamma correction is not performed when mipmap data provided. (only if inputGamma != outputGamma)
// @@ This code is too complicated, too prone to erros, and hard to understand. Must be simplified! // @@ This code is too complicated, too prone to erros, and hard to understand. Must be simplified!
quantize(img, inputOptions.m, format);
compressMipmap(img, outputOptions, compressionOptions.m);
// @@ Quantization should be done in compressMipmap!
quantize(img, inputOptions, format);
compressMipmap(img, outputOptions, compressionOptions);
if (img != mipmap.data) if (img != mipmap.data)
{ {
delete img; delete img;
} }
if (!inputOptions.m.generateMipmaps || (inputOptions.m.maxLevel >= 0 && m >= inputOptions.m.maxLevel)) { // Compute extents of next mipmap:
// continue with next face. w = max(1U, w / 2);
break; h = max(1U, h / 2);
} d = max(1U, d / 2);
} }
*/
} }
outputOptions.closeFile();
return true; return true;
} }
/// Compress the input texture with the given compression options.
bool nvtt::compress(const InputOptions & inputOptions, const OutputOptions & outputOptions, const CompressionOptions & compressionOptions)
{
// @@ Hack this is necessary because of the pimpl transition.
initOptions(const_cast<OutputOptions *>(&outputOptions));
return ::compress(inputOptions.m, outputOptions.m, compressionOptions.m);
}
/// Estimate the size of compressing the input with the given options. /// Estimate the size of compressing the input with the given options.
int nvtt::estimateSize(const InputOptions & inputOptions, const CompressionOptions & compressionOptions) int nvtt::estimateSize(const InputOptions & inputOptions, const CompressionOptions & compressionOptions)
{ {
Format format = compressionOptions.m.format; const Format format = compressionOptions.m.format;
const uint bitCount = compressionOptions.m.bitcount; const uint bitCount = compressionOptions.m.bitcount;
inputOptions.m.computeTargetExtents();
uint mipmapCount = inputOptions.m.realMipmapCount();
int size = 0; int size = 0;
for (int f = 0; f < inputOptions.m.faceCount; f++) for (uint f = 0; f < inputOptions.m.faceCount; f++)
{ {
for (int m = 0; m < inputOptions.m.mipmapCount; m++) uint w = inputOptions.m.targetWidth;
uint h = inputOptions.m.targetHeight;
uint d = inputOptions.m.targetDepth;
for (uint m = 0; m < mipmapCount; m++)
{ {
int idx = f * inputOptions.m.mipmapCount + m; size += computeImageSize(w, h, bitCount, format);
const InputOptions::Private::Image & img = inputOptions.m.images[idx];
size += computeImageSize(img.width, img.height, bitCount, format); // Compute extents of next mipmap:
w = max(1U, w / 2);
if (!inputOptions.m.generateMipmaps || (inputOptions.m.maxLevel >= 0 && m >= inputOptions.m.maxLevel)) { h = max(1U, h / 2);
// continue with next face. d = max(1U, d / 2);
break;
}
} }
} }
@ -530,9 +799,15 @@ const char * nvtt::errorString(Error e)
return "Unsupported feature"; return "Unsupported feature";
case Error_CudaError: case Error_CudaError:
return "CUDA error"; return "CUDA error";
case Error_FileOpen:
return "Error opening file";
case Error_FileWrite:
return "Error writing through output handler";
case Error_Unknown: case Error_Unknown:
default: //default:
return "Unknown error"; return "Unknown error";
} }
return "Invalid error";
} }

View File

@ -40,6 +40,9 @@
#define NVTT_CLASS #define NVTT_CLASS
#endif #endif
#define NVTT_DEPRECATED NVTT_API NV_DEPRECATED
// Public interface. // Public interface.
namespace nvtt namespace nvtt
{ {
@ -49,7 +52,7 @@ namespace nvtt
// No compression. // No compression.
Format_RGB, Format_RGB,
Format_RGBA = Format_RGB, Format_RGBA = Format_RGB,
// DX9 formats. // DX9 formats.
Format_DXT1, Format_DXT1,
Format_DXT1a, // DXT1 with binary alpha. Format_DXT1a, // DXT1 with binary alpha.
@ -65,9 +68,6 @@ namespace nvtt
Format_BC3n = Format_DXT5n, Format_BC3n = Format_DXT5n,
Format_BC4, // ATI1 Format_BC4, // ATI1
Format_BC5, // 3DC, ATI2 Format_BC5, // 3DC, ATI2
// OpenGL formats.
Format_LATC = Format_BC5,
}; };
/// Quality modes. /// Quality modes.
@ -91,7 +91,9 @@ namespace nvtt
NVTT_API void setFormat(Format format); NVTT_API void setFormat(Format format);
NVTT_API void setQuality(Quality quality, float errorThreshold = 0.5f); NVTT_API void setQuality(Quality quality, float errorThreshold = 0.5f);
NVTT_API void setColorWeights(float red, float green, float blue); NVTT_API void setColorWeights(float red, float green, float blue);
NVTT_API void enableHardwareCompression(bool enable);
NVTT_DEPRECATED void enableHardwareCompression(bool enable);
NVTT_API void enableCudaCompression(bool enable);
NVTT_API void setExternalCompressor(const char * name); NVTT_API void setExternalCompressor(const char * name);
@ -136,12 +138,28 @@ namespace nvtt
MipmapFilter_Kaiser, ///< Kaiser-windowed Sinc filter is the best downsampling filter. MipmapFilter_Kaiser, ///< Kaiser-windowed Sinc filter is the best downsampling filter.
}; };
/// Color transformation.
enum ColorTransform enum ColorTransform
{ {
ColorTransform_None, ColorTransform_None,
ColorTransform_Linear, ColorTransform_Linear,
ColorTransform_CoYCg, };
ColorTransform_CoSCgY,
/// Extents rounding mode.
enum RoundMode
{
RoundMode_None,
RoundMode_ToNextPowerOfTwo,
RoundMode_ToNearestPowerOfTwo,
RoundMode_ToPreviousPowerOfTwo,
};
/// Alpha mode.
enum AlphaMode
{
AlphaMode_None,
AlphaMode_Transparency,
AlphaMode_Premultiplied,
}; };
/// Input options. Specify format and layout of the input texture. /// Input options. Specify format and layout of the input texture.
@ -156,37 +174,45 @@ namespace nvtt
// Setup input layout. // Setup input layout.
NVTT_API void setTextureLayout(TextureType type, int w, int h, int d = 1); NVTT_API void setTextureLayout(TextureType type, int w, int h, int d = 1);
NVTT_API void resetTextureLayout(); NVTT_API void resetTextureLayout();
// Set mipmap data. Copies the data. // Set mipmap data. Copies the data.
NVTT_API bool setMipmapData(const void * data, int w, int h, int d = 1, int face = 0, int mipmap = 0); NVTT_API bool setMipmapData(const void * data, int w, int h, int d = 1, int face = 0, int mipmap = 0);
// Describe the format of the input. // Describe the format of the input.
NVTT_API void setFormat(InputFormat fmt, bool alphaTransparency); NVTT_DEPRECATED void setFormat(InputFormat format, bool alphaTransparency);
NVTT_API void setFormat(InputFormat format);
// Set the way the input alpha channel is interpreted.
NVTT_API void setAlphaMode(AlphaMode alphaMode);
// Set gamma settings. // Set gamma settings.
NVTT_API void setGamma(float inputGamma, float outputGamma); NVTT_API void setGamma(float inputGamma, float outputGamma);
// Set texture wrappign mode. // Set texture wrappign mode.
NVTT_API void setWrapMode(WrapMode mode); NVTT_API void setWrapMode(WrapMode mode);
// Set mipmapping options. // Set mipmapping options.
NVTT_API void setMipmapping(bool generateMipmaps, MipmapFilter filter = MipmapFilter_Box, int maxLevel = -1); NVTT_API void setMipmapping(bool generateMipmaps, MipmapFilter filter = MipmapFilter_Box, int maxLevel = -1);
NVTT_API void setKaiserParameters(int width, float alpha, float stretch); NVTT_API void setKaiserParameters(float width, float alpha, float stretch);
// Set quantization options. // Set quantization options.
NVTT_API void setQuantization(bool colorDithering, bool alphaDithering, bool binaryAlpha, int alphaThreshold = 127); NVTT_DEPRECATED void setQuantization(bool colorDithering, bool alphaDithering, bool binaryAlpha, int alphaThreshold = 127);
// Set normal map options. // Set normal map options.
NVTT_API void setNormalMap(bool b); NVTT_API void setNormalMap(bool b);
NVTT_API void setConvertToNormalMap(bool convert); NVTT_API void setConvertToNormalMap(bool convert);
NVTT_API void setHeightEvaluation(float redScale, float greenScale, float blueScale, float alphaScale); NVTT_API void setHeightEvaluation(float redScale, float greenScale, float blueScale, float alphaScale);
NVTT_API void setNormalFilter(float small, float medium, float big, float large); NVTT_API void setNormalFilter(float small, float medium, float big, float large);
NVTT_API void setNormalizeMipmaps(bool b); NVTT_API void setNormalizeMipmaps(bool b);
// Set color transforms. // Set color transforms.
NVTT_API void setColorTransform(ColorTransform t); NVTT_API void setColorTransform(ColorTransform t);
NVTT_API void setLinearTransfrom(int channel, float w0, float w1, float w2, float w3); NVTT_API void setLinearTransfrom(int channel, float w0, float w1, float w2, float w3);
// Set resizing options.
NVTT_API void setMaxExtents(int d);
NVTT_API void setRoundMode(RoundMode mode);
//private: //private:
struct Private; struct Private;
Private & m; Private & m;
@ -213,6 +239,8 @@ namespace nvtt
Error_UnsupportedFeature, Error_UnsupportedFeature,
Error_CudaError, Error_CudaError,
Error_Unknown, Error_Unknown,
Error_FileOpen,
Error_FileWrite,
}; };
/// Error handler. /// Error handler.
@ -229,15 +257,26 @@ namespace nvtt
/// the compressor to the user. /// the compressor to the user.
struct OutputOptions struct OutputOptions
{ {
OutputOptions() : outputHandler(NULL), outputHeader(true) { reset(); } NVTT_API OutputOptions();
OutputOptions(OutputHandler * oh, ErrorHandler * eh) : outputHandler(oh), errorHandler(eh), outputHeader(true) { reset(); } NVTT_DEPRECATED OutputOptions(OutputHandler * oh, ErrorHandler * eh);
NVTT_API ~OutputOptions();
// Set default options. // Set default options.
NVTT_API void reset(); NVTT_API void reset();
OutputHandler * outputHandler; NVTT_API void setFileName(const char * fileName);
ErrorHandler * errorHandler;
bool outputHeader; NVTT_API void setOutputHandler(OutputHandler * outputHandler);
NVTT_API void setErrorHandler(ErrorHandler * errorHandler);
NVTT_API void setOutputHeader(bool outputHeader);
NVTT_DEPRECATED OutputHandler * outputHandler;
NVTT_DEPRECATED ErrorHandler * errorHandler;
NVTT_DEPRECATED bool outputHeader;
//private:
struct Private;
Private & m;
}; };