Skip to content

Commit

Permalink
common: Rewrote image conversion, other misc improovements/refactors
Browse files Browse the repository at this point in the history
  • Loading branch information
JJL772 committed Aug 31, 2024
1 parent 8d02e5a commit c736fab
Show file tree
Hide file tree
Showing 9 changed files with 428 additions and 107 deletions.
28 changes: 28 additions & 0 deletions CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ set(CMAKE_CXX_STANDARD 20)

# Project settings
option(BUILD_GUI "Build the VTFViewer GUI" ON)
option(BUILD_TESTS "Build test binaries" ON)

# Global flags, mainly for UNIX. Use $ORIGIN rpath & -fPIC
set(CMAKE_POSITION_INDEPENDENT_CODE ON)
Expand All @@ -18,6 +19,7 @@ set(CMAKE_MSVC_RUNTIME_LIBRARY "MultiThreaded$<$<CONFIG:Debug>:Debug>")
# Build vtflib as static lib
add_subdirectory(external/vtflib)
add_subdirectory(external/fmtlib)
add_subdirectory(external/gtest)

add_definitions(-DVTFLIB_STATIC=1)

Expand Down Expand Up @@ -99,6 +101,32 @@ if (BUILD_GUI)
)
endif()

##############################
# Tests
##############################

if (BUILD_TESTS)
add_executable(
tests

src/tests/image_tests.cpp
)

target_link_libraries(
tests PRIVATE

gtest
vtflib_static
com
)

target_include_directories(
tests PRIVATE

src
)
endif()

##############################
# Version header
##############################
Expand Down
16 changes: 12 additions & 4 deletions src/cli/action_convert.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -319,15 +319,15 @@ bool ActionConvert::process_file(
std::max(formatInfo.uiRedBitsPerPixel, formatInfo.uiGreenBitsPerPixel),
std::max(formatInfo.uiBlueBitsPerPixel, formatInfo.uiAlphaBitsPerPixel));
if (maxBpp > 16) {
procChanType = imglib::Float;
procChanType = imglib::ChannelType::Float;
return IMAGE_FORMAT_RGBA32323232F;
}
else if (maxBpp > 8) {
procChanType = imglib::UInt16;
return IMAGE_FORMAT_RGBA16161616F;
procChanType = imglib::ChannelType::UInt16;
return IMAGE_FORMAT_RGBA16161616;
}
else {
procChanType = imglib::UInt8;
procChanType = imglib::ChannelType::UInt8;
return IMAGE_FORMAT_RGBA8888;
}
}();
Expand Down Expand Up @@ -521,6 +521,14 @@ bool ActionConvert::add_image_data(
return false;
}

// Hack for VTFLib; Ensure we have an alpha channel because that's well supported in that horrible code
if (image->channels() < 4 && image->type() != imglib::ChannelType::UInt8) {
if (!image->convert(image->type(), 4)) {
std::cerr << fmt::format("Failed to convert {}\n", imageSrc.c_str());
return false;
}
}

// Add the raw image data
return add_image_data_raw(
file, image->data(), format, image->vtf_format(), image->width(), image->height(), create);
Expand Down
2 changes: 1 addition & 1 deletion src/cli/action_extract.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -217,7 +217,7 @@ bool ActionExtract::extract_file(
return false;
}

imglib::Image image(imageData, destIsFloat ? imglib::Float : imglib::UInt8, comps, w, h, true);
imglib::Image image(imageData, destIsFloat ? imglib::ChannelType::Float : imglib::ChannelType::UInt8, comps, w, h, true);
if (!image.save(outFile.string().c_str(), targetFmt)) {
std::cerr << fmt::format("Could not save image to '{}'!\n", outFile.string());
return false;
Expand Down
35 changes: 22 additions & 13 deletions src/cli/action_pack.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -212,7 +212,7 @@ int ActionPack::exec(const OptionList& opts) {
//
static std::shared_ptr<imglib::Image> load_image(const std::filesystem::path& path) {

auto img = imglib::Image::load(path);
auto img = imglib::Image::load(path); // Force UInt8 for packed images
if (!img)
std::cerr << fmt::format("Could not load image '{}'\n", path.string());
return img;
Expand All @@ -236,20 +236,23 @@ static void determine_size(int* w, int* h, const std::shared_ptr<imglib::Image>
//
// Resize images if required and converts too!
//
static void resize_if_required(const std::shared_ptr<imglib::Image>& image, int w, int h) {
static bool resize_if_required(const std::shared_ptr<imglib::Image>& image, int w, int h) {
if (!image)
return;
return true;

// @TODO: For now we're just going to force 8 bit per channel.
// Sometimes we do get 16bpc images, mainly for height data, but we're cramming that into a RGBA8888 texture
// anyways. It'd be best to eventually support RGBA16F normals for instances where you need precise height data.
if (image->type() != imglib::UInt8) {
assert(image->convert(imglib::UInt8));
if (image->type() != imglib::ChannelType::UInt8) {
if (!image->convert(imglib::ChannelType::UInt8)) {
std::cerr << "Failed to convert image\n";
return false;
}
}

if (image->width() == w && image->height() == h)
return;
assert(image->resize(w, h));
return true;
return image->resize(w, h);
}

//
Expand Down Expand Up @@ -296,10 +299,14 @@ bool ActionPack::pack_mrao(
}

// Resize images if required
resize_if_required(roughnessData, w, h);
resize_if_required(metalnessData, w, h);
resize_if_required(aoData, w, h);
resize_if_required(tmaskData, w, h);
if (!resize_if_required(roughnessData, w, h))
return false;
if (!resize_if_required(metalnessData, w, h))
return false;
if (!resize_if_required(aoData, w, h))
return false;
if (!resize_if_required(tmaskData, w, h))
return false;

// Packing config
pack::ChannelPack_t pack[] = {
Expand Down Expand Up @@ -400,8 +407,10 @@ bool ActionPack::pack_normal(
}

// Resize images if required
resize_if_required(normalData, w, h);
resize_if_required(heightData, w, h);
if (!resize_if_required(normalData, w, h))
return false;
if (!resize_if_required(heightData, w, h))
return false;

// Convert normal to DX if necessary
if (isGL)
Expand Down
98 changes: 65 additions & 33 deletions src/common/image.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@
#include "image.hpp"
#include "util.hpp"
#include "strtools.hpp"
#include "lwiconv.hpp"

#include <cstring>
#include <cassert>
Expand Down Expand Up @@ -52,16 +53,16 @@ Image::~Image() {
free(m_data);
}

std::shared_ptr<Image> Image::load(const char* path) {
std::shared_ptr<Image> Image::load(const char* path, ChannelType convertOnLoad) {
FILE* fp = fopen(path, "rb");
if (!fp)
return nullptr;
auto img = load(fp);
auto img = load(fp, convertOnLoad);
fclose(fp);
return img;
}

std::shared_ptr<Image> Image::load(FILE* fp) {
std::shared_ptr<Image> Image::load(FILE* fp, ChannelType convertOnLoad) {
auto info = image_info(fp);
auto image = std::make_shared<Image>();
if (info.type == ChannelType::Float) {
Expand All @@ -73,9 +74,15 @@ std::shared_ptr<Image> Image::load(FILE* fp) {
else {
image->m_data = stbi_load_from_file(fp, &image->m_width, &image->m_height, &image->m_comps, info.comps);
}
image->m_type = info.type;

if (!image->m_data)
return nullptr;

if (convertOnLoad != ChannelType::None && convertOnLoad != info.type)
if (!image->convert(convertOnLoad))
return nullptr; // Convert on load failed

return image;
}

Expand All @@ -86,6 +93,8 @@ void Image::clear() {
}

bool Image::save(const char* file, FileFormat format) {
using enum ChannelType;

if (!file || !m_data || (format != Tga && format != Png && format != Jpeg && format != Bmp && format != Hdr))
return false;

Expand All @@ -97,7 +106,7 @@ bool Image::save(const char* file, FileFormat format) {
if (m_type != Float) {
dataToUse = malloc(m_width * m_height * sizeof(float) * m_comps);
dataIsOurs = true;
if (!convert_formats(m_data, dataToUse, m_type, Float, m_comps, m_width, m_height)) {
if (!convert_formats(m_data, dataToUse, m_type, Float, m_width, m_height, m_comps, m_comps, pixel_size(), imglib::pixel_size(Float, m_comps))) {
free(dataToUse);
return false;
}
Expand All @@ -115,7 +124,7 @@ bool Image::save(const char* file, FileFormat format) {
if (m_type != UInt8) {
dataToUse = malloc(m_width * m_height * sizeof(uint8_t) * m_comps);
dataIsOurs = true;
if (!convert_formats(m_data, dataToUse, m_type, UInt8, m_comps, m_width, m_height)) {
if (!convert_formats(m_data, dataToUse, m_type, UInt8, m_width, m_height, m_comps, m_comps, pixel_size(), imglib::pixel_size(UInt8, m_comps))) {
free(dataToUse);
return false;
}
Expand Down Expand Up @@ -157,10 +166,10 @@ bool Image::resize(int newW, int newH) {

VTFImageFormat Image::vtf_format() const {
switch (m_type) {
case imglib::UInt16:
// @TODO: How to handle RGBA16? DONT i guess
return IMAGE_FORMAT_RGBA16161616F;
case imglib::Float:
case ChannelType::UInt16:
// @TODO: How to handle RGB16? DONT i guess
return IMAGE_FORMAT_RGBA16161616;
case ChannelType::Float:
return (m_comps == 3) ? IMAGE_FORMAT_RGB323232F
: (m_comps == 1 ? IMAGE_FORMAT_R32F : IMAGE_FORMAT_RGBA32323232F);
default:
Expand All @@ -170,6 +179,7 @@ VTFImageFormat Image::vtf_format() const {

bool imglib::resize(
void* indata, void** useroutdata, ChannelType srcType, int comps, int w, int h, int newW, int newH) {
using enum ChannelType;
stbir_datatype type;
switch (srcType) {
case Float:
Expand Down Expand Up @@ -215,63 +225,66 @@ size_t imglib::bytes_for_image(int w, int h, ChannelType type, int comps) {
return w * h * comps * bpc;
}

template <int COMPS>
bool convert_formats_internal(
const void* srcData, void* dstData, ChannelType srcChanType, ChannelType dstChanType, int w, int h) {
static_assert(COMPS > 0 && COMPS <= 4, "Comps is out of range");
const void* srcData, void* dstData, ChannelType srcChanType, ChannelType dstChanType, int w, int h, int inComps, int outComps, int inStride, int outStride, const lwiconv::PixelF& pdef) {
using enum ChannelType;

if (srcChanType == UInt8) {
// RGBX32
if (dstChanType == Float)
convert_8_to_32<COMPS>(srcData, dstData, w, h);
lwiconv::convert_generic<uint8_t, float>(srcData, dstData, w, h, inComps, outComps, inStride, outStride, pdef);
// RGBX16
else if (dstChanType == UInt16)
convert_8_to_16<COMPS>(srcData, dstData, w, h);
lwiconv::convert_generic<uint8_t, uint16_t>(srcData, dstData, w, h, inComps, outComps, inStride, outStride, pdef);
// RGBX8 (just adding/removing channel(s))
else if (dstChanType == UInt8)
lwiconv::convert_generic<uint8_t, uint8_t>(srcData, dstData, w, h, inComps, outComps, inStride, outStride, pdef);
return true;
}
// RGBX32 -> RGBX[8|16]
else if (srcChanType == Float) {
// RGBX16
if (dstChanType == UInt16)
convert_32_to_16<COMPS>(srcData, dstData, w, h);
lwiconv::convert_generic<float, uint16_t>(srcData, dstData, w, h, inComps, outComps, inStride, outStride, pdef);
// RGBX8
else if (dstChanType == UInt8)
convert_32_to_8<COMPS>(srcData, dstData, w, h);
lwiconv::convert_generic<float, uint8_t>(srcData, dstData, w, h, inComps, outComps, inStride, outStride, pdef);
// RGBX32 (just adding/removing channel(s))
else if (dstChanType == Float)
lwiconv::convert_generic<float, float>(srcData, dstData, w, h, inComps, outComps, inStride, outStride, pdef);
return true;
}
// RGBX16
// RGBX16 -> RGBX[8|32F]
else if (srcChanType == UInt16) {
// RGBX8
if (dstChanType == UInt8)
convert_16_to_8<COMPS>(srcData, dstData, w, h);
lwiconv::convert_generic<uint16_t, uint8_t>(srcData, dstData, w, h, inComps, outComps, inStride, outStride, pdef);
// RGBX32
else if (dstChanType == Float)
convert_16_to_32<COMPS>(srcData, dstData, w, h);
lwiconv::convert_generic<uint16_t, float>(srcData, dstData, w, h, inComps, outComps, inStride, outStride, pdef);
// RGBX16 (just adding/removing channel(s))
else if (dstChanType == UInt16)
lwiconv::convert_generic<uint16_t, uint16_t>(srcData, dstData, w, h, inComps, outComps, inStride, outStride, pdef);
return true;
}

return false;
}

bool imglib::convert_formats(
const void* srcData, void* dstData, ChannelType srcChanType, ChannelType dstChanType, int comps, int w, int h) {
const void* srcData, void* dstData, ChannelType srcChanType, ChannelType dstChanType, int w, int h, int inComps, int outComps, int inStride, int outStride, const lwiconv::PixelF& pdef) {
// No conv needed
if (srcChanType == dstChanType)
if (srcChanType == dstChanType && inComps == outComps)
return true;

if (comps == 4)
return convert_formats_internal<4>(srcData, dstData, srcChanType, dstChanType, w, h);
else if (comps == 3)
return convert_formats_internal<3>(srcData, dstData, srcChanType, dstChanType, w, h);
else if (comps == 2)
return convert_formats_internal<2>(srcData, dstData, srcChanType, dstChanType, w, h);
else if (comps == 1)
return convert_formats_internal<1>(srcData, dstData, srcChanType, dstChanType, w, h);
return false;
return convert_formats_internal(srcData, dstData, srcChanType, dstChanType, w, h, inComps, outComps, inStride, outStride, pdef);
}

bool Image::convert(ChannelType dstChanType) {
void* dst = malloc(imglib::bytes_for_image(m_width, m_height, m_type, m_comps));
bool Image::convert(ChannelType dstChanType, int channels, const lwiconv::PixelF& pdef) {
channels = channels <= 0 ? m_comps : channels;
void* dst = malloc(imglib::bytes_for_image(m_width, m_height, m_type, channels));

if (!convert_formats(m_data, dst, m_type, dstChanType, m_comps, m_width, m_height)) {
if (!convert_formats(m_data, dst, m_type, dstChanType, m_width, m_height, m_comps, channels, pixel_size(), channels * channel_size(dstChanType), pdef)) {
free(dst);
return false;
}
Expand Down Expand Up @@ -302,6 +315,7 @@ static bool process_image_internal(void* indata, int comps, int w, int h, ProcFl
}

bool Image::process(ProcFlags flags) {
using enum ChannelType;
switch (m_type) {
case UInt8:
return process_image_internal<uint8_t>(m_data, m_comps, m_width, m_height, flags);
Expand Down Expand Up @@ -374,3 +388,21 @@ static ImageInfo_t image_info(FILE* fp) {

return info;
}

size_t imglib::pixel_size(ChannelType type, int channels) {
return channels * channel_size(type);
}

size_t imglib::channel_size(ChannelType type) {
switch(type) {
case imglib::ChannelType::UInt8:
return 1;
case imglib::ChannelType::UInt16:
return 2;
case imglib::ChannelType::Float:
return 4;
default:
assert(0);
return 1;
}
}
Loading

0 comments on commit c736fab

Please sign in to comment.