blob: bd7449609a74ce02f97285b4d5b16d5063e61148 [file] [log] [blame]
// SPDX-License-Identifier: Apache-2.0
// ----------------------------------------------------------------------------
// Copyright 2011-2022 Arm Limited
//
// Licensed under the Apache License, Version 2.0 (the "License"); you may not
// use this file except in compliance with the License. You may obtain a copy
// of the License at:
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
// License for the specific language governing permissions and limitations
// under the License.
// ----------------------------------------------------------------------------
/**
* @brief Functions for building the implementation of stb_image and tinyexr.
*/
#include <cstdlib>
#include <cstdio>
#include <fstream>
#include <vector>
#include "astcenccli_internal.h"
// Configure the STB image imagewrite library build.
#define STB_IMAGE_IMPLEMENTATION
#define STB_IMAGE_WRITE_IMPLEMENTATION
#define STBI_NO_GIF
#define STBI_NO_PIC
#define STBI_NO_PNM
#define STBI_NO_PNG
#define STBI_NO_PSD
// Configure the TinyEXR library build.
#define TINYEXR_IMPLEMENTATION
// Configure the Wuffs library build.
#define WUFFS_IMPLEMENTATION
#define WUFFS_CONFIG__MODULES
#define WUFFS_CONFIG__MODULE__ADLER32
#define WUFFS_CONFIG__MODULE__BASE
#define WUFFS_CONFIG__MODULE__CRC32
#define WUFFS_CONFIG__MODULE__DEFLATE
#define WUFFS_CONFIG__MODULE__PNG
#define WUFFS_CONFIG__MODULE__ZLIB
#include "wuffs-v0.3.c"
// For both libraries force asserts (which can be triggered by corrupt input
// images) to be handled at runtime in release builds to avoid security issues.
#define STBI_ASSERT(x) astcenc_runtime_assert(x)
#define TEXR_ASSERT(x) astcenc_runtime_assert(x)
/**
* @brief Trap image load failures and convert into a runtime error.
*/
static void astcenc_runtime_assert(bool condition)
{
if (!condition)
{
printf("ERROR: Corrupt input image\n");
exit(1);
}
}
#include "stb_image.h"
#include "stb_image_write.h"
#include "tinyexr.h"
/**
* @brief Load an image using Wuffs to provide the loader.
*
* @param filename The name of the file to load.
* @param y_flip Should the image be vertically flipped?
* @param[out] is_hdr Is this an HDR image load?
* @param[out] component_count The number of components in the data.
*
* @return The loaded image data in a canonical 4 channel format, or @c nullptr on error.
*/
astcenc_image* load_png_with_wuffs(
const char* filename,
bool y_flip,
bool& is_hdr,
unsigned int& component_count
) {
is_hdr = false;
component_count = 4;
std::ifstream file(filename, std::ios::binary | std::ios::ate);
if (!file)
{
printf("ERROR: Failed to load image %s (can't fopen)\n", filename);
return nullptr;
}
std::streamsize size = file.tellg();
file.seekg(0, std::ios::beg);
std::vector<uint8_t> buffer(size);
file.read((char*)buffer.data(), size);
wuffs_png__decoder *dec = wuffs_png__decoder__alloc();
if (!dec)
{
return nullptr;
}
wuffs_base__image_config ic;
wuffs_base__io_buffer src = wuffs_base__ptr_u8__reader(buffer.data(), size, true);
wuffs_base__status status = wuffs_png__decoder__decode_image_config(dec, &ic, &src);
if (status.repr)
{
return nullptr;
}
uint32_t dim_x = wuffs_base__pixel_config__width(&ic.pixcfg);
uint32_t dim_y = wuffs_base__pixel_config__height(&ic.pixcfg);
size_t num_pixels = dim_x * dim_y;
if (num_pixels > (SIZE_MAX / 4))
{
return nullptr;
}
// Override the image's native pixel format to be RGBA_NONPREMUL
wuffs_base__pixel_config__set(
&ic.pixcfg,
WUFFS_BASE__PIXEL_FORMAT__RGBA_NONPREMUL,
WUFFS_BASE__PIXEL_SUBSAMPLING__NONE,
dim_x, dim_y);
// Configure the work buffer
size_t workbuf_len = wuffs_png__decoder__workbuf_len(dec).max_incl;
if (workbuf_len > SIZE_MAX)
{
return nullptr;
}
wuffs_base__slice_u8 workbuf_slice = wuffs_base__make_slice_u8((uint8_t*)malloc(workbuf_len), workbuf_len);
if (!workbuf_slice.ptr)
{
return nullptr;
}
wuffs_base__slice_u8 pixbuf_slice = wuffs_base__make_slice_u8((uint8_t*)malloc(num_pixels * 4), num_pixels * 4);
if (!pixbuf_slice.ptr)
{
return nullptr;
}
wuffs_base__pixel_buffer pb;
status = wuffs_base__pixel_buffer__set_from_slice(&pb, &ic.pixcfg, pixbuf_slice);
if (status.repr)
{
return nullptr;
}
// Decode the pixels
status = wuffs_png__decoder__decode_frame(dec, &pb, &src, WUFFS_BASE__PIXEL_BLEND__SRC, workbuf_slice, NULL);
if (status.repr)
{
return nullptr;
}
astcenc_image* img = astc_img_from_unorm8x4_array(pixbuf_slice.ptr, dim_x, dim_y, y_flip);
free(pixbuf_slice.ptr);
free(workbuf_slice.ptr);
free(dec);
return img;
}