// 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 creating in-memory ASTC image structures. */ #include #include #include "astcenccli_internal.h" /* See header for documentation. */ astcenc_image *alloc_image( unsigned int bitness, unsigned int dim_x, unsigned int dim_y, unsigned int dim_z ) { astcenc_image *img = new astcenc_image; img->dim_x = dim_x; img->dim_y = dim_y; img->dim_z = dim_z; void** data = new void*[dim_z]; img->data = data; if (bitness == 8) { img->data_type = ASTCENC_TYPE_U8; for (unsigned int z = 0; z < dim_z; z++) { data[z] = new uint8_t[dim_x * dim_y * 4]; } } else if (bitness == 16) { img->data_type = ASTCENC_TYPE_F16; for (unsigned int z = 0; z < dim_z; z++) { data[z] = new uint16_t[dim_x * dim_y * 4]; } } else // if (bitness == 32) { assert(bitness == 32); img->data_type = ASTCENC_TYPE_F32; for (unsigned int z = 0; z < dim_z; z++) { data[z] = new float[dim_x * dim_y * 4]; } } return img; } /* See header for documentation. */ void free_image(astcenc_image * img) { if (img == nullptr) { return; } for (unsigned int z = 0; z < img->dim_z; z++) { delete[] reinterpret_cast(img->data[z]); } delete[] img->data; delete img; } /* See header for documentation. */ int determine_image_components(const astcenc_image * img) { unsigned int dim_x = img->dim_x; unsigned int dim_y = img->dim_y; unsigned int dim_z = img->dim_z; // Scan through the image data to determine how many color components the image has bool is_luma = true; bool has_alpha = false; if (img->data_type == ASTCENC_TYPE_U8) { for (unsigned int z = 0; z < dim_z; z++) { uint8_t* data8 = static_cast(img->data[z]); for (unsigned int y = 0; y < dim_y; y++) { for (unsigned int x = 0; x < dim_x; x++) { int r = data8[(4 * dim_x * y) + (4 * x )]; int g = data8[(4 * dim_x * y) + (4 * x + 1)]; int b = data8[(4 * dim_x * y) + (4 * x + 2)]; int a = data8[(4 * dim_x * y) + (4 * x + 3)]; is_luma = is_luma && (r == g) && (r == b); has_alpha = has_alpha || (a != 0xFF); } } } } else if (img->data_type == ASTCENC_TYPE_F16) { for (unsigned int z = 0; z < dim_z; z++) { uint16_t* data16 = static_cast(img->data[z]); for (unsigned int y = 0; y < dim_y; y++) { for (unsigned int x = 0; x < dim_x; x++) { int r = data16[(4 * dim_x * y) + (4 * x )]; int g = data16[(4 * dim_x * y) + (4 * x + 1)]; int b = data16[(4 * dim_x * y) + (4 * x + 2)]; int a = data16[(4 * dim_x * y) + (4 * x + 3)]; is_luma = is_luma && (r == g) && (r == b); has_alpha = has_alpha || ((a ^ 0xC3FF) != 0xFFFF); // a ^ 0xC3FF returns FFFF if and only if the input is 1.0 } } } } else // if (img->data_type == ASTCENC_TYPE_F32) { assert(img->data_type == ASTCENC_TYPE_F32); for (unsigned int z = 0; z < dim_z; z++) { float* data32 = static_cast(img->data[z]); for (unsigned int y = 0; y < dim_y; y++) { for (unsigned int x = 0; x < dim_x; x++) { float r = data32[(4 * dim_x * y) + (4 * x )]; float g = data32[(4 * dim_x * y) + (4 * x + 1)]; float b = data32[(4 * dim_x * y) + (4 * x + 2)]; float a = data32[(4 * dim_x * y) + (4 * x + 3)]; is_luma = is_luma && (r == g) && (r == b); has_alpha = has_alpha || (a != 1.0f); } } } } int image_components = 1 + (is_luma == 0 ? 2 : 0) + (has_alpha ? 1 : 0); return image_components; } /* See header for documentation. */ astcenc_image* astc_img_from_floatx4_array( const float* data, unsigned int dim_x, unsigned int dim_y, bool y_flip ) { astcenc_image* img = alloc_image(16, dim_x, dim_y, 1); for (unsigned int y = 0; y < dim_y; y++) { uint16_t* data16 = static_cast(img->data[0]); unsigned int y_src = y_flip ? (dim_y - y - 1) : y; const float* src = data + 4 * dim_x * y_src; for (unsigned int x = 0; x < dim_x; x++) { vint4 colorf16 = float_to_float16(vfloat4( src[4 * x ], src[4 * x + 1], src[4 * x + 2], src[4 * x + 3] )); data16[(4 * dim_x * y) + (4 * x )] = static_cast(colorf16.lane<0>()); data16[(4 * dim_x * y) + (4 * x + 1)] = static_cast(colorf16.lane<1>()); data16[(4 * dim_x * y) + (4 * x + 2)] = static_cast(colorf16.lane<2>()); data16[(4 * dim_x * y) + (4 * x + 3)] = static_cast(colorf16.lane<3>()); } } return img; } /* See header for documentation. */ astcenc_image* astc_img_from_unorm8x4_array( const uint8_t* data, unsigned int dim_x, unsigned int dim_y, bool y_flip ) { astcenc_image* img = alloc_image(8, dim_x, dim_y, 1); for (unsigned int y = 0; y < dim_y; y++) { uint8_t* data8 = static_cast(img->data[0]); unsigned int y_src = y_flip ? (dim_y - y - 1) : y; const uint8_t* src = data + 4 * dim_x * y_src; for (unsigned int x = 0; x < dim_x; x++) { data8[(4 * dim_x * y) + (4 * x )] = src[4 * x ]; data8[(4 * dim_x * y) + (4 * x + 1)] = src[4 * x + 1]; data8[(4 * dim_x * y) + (4 * x + 2)] = src[4 * x + 2]; data8[(4 * dim_x * y) + (4 * x + 3)] = src[4 * x + 3]; } } return img; } // initialize a flattened array of float values from an ASTC codec image // The returned array is allocated with new[] and must be deleted with delete[]. /* See header for documentation. */ float* floatx4_array_from_astc_img( const astcenc_image* img, bool y_flip ) { unsigned int dim_x = img->dim_x; unsigned int dim_y = img->dim_y; float *buf = new float[4 * dim_x * dim_y]; if (img->data_type == ASTCENC_TYPE_U8) { uint8_t* data8 = static_cast(img->data[0]); for (unsigned int y = 0; y < dim_y; y++) { unsigned int ymod = y_flip ? dim_y - y - 1 : y; float* dst = buf + y * dim_x * 4; for (unsigned int x = 0; x < dim_x; x++) { dst[4 * x ] = data8[(4 * dim_x * ymod) + (4 * x )] * (1.0f / 255.0f); dst[4 * x + 1] = data8[(4 * dim_x * ymod) + (4 * x + 1)] * (1.0f / 255.0f); dst[4 * x + 2] = data8[(4 * dim_x * ymod) + (4 * x + 2)] * (1.0f / 255.0f); dst[4 * x + 3] = data8[(4 * dim_x * ymod) + (4 * x + 3)] * (1.0f / 255.0f); } } } else if (img->data_type == ASTCENC_TYPE_F16) { uint16_t* data16 = static_cast(img->data[0]); for (unsigned int y = 0; y < dim_y; y++) { unsigned int ymod = y_flip ? dim_y - y - 1 : y; float *dst = buf + y * dim_x * 4; for (unsigned int x = 0; x < dim_x; x++) { vint4 colori( data16[(4 * dim_x * ymod) + (4 * x )], data16[(4 * dim_x * ymod) + (4 * x + 1)], data16[(4 * dim_x * ymod) + (4 * x + 2)], data16[(4 * dim_x * ymod) + (4 * x + 3)] ); vfloat4 color = float16_to_float(colori); store(color, dst + 4 * x); } } } else // if (img->data_type == ASTCENC_TYPE_F32) { assert(img->data_type == ASTCENC_TYPE_F32); float* data32 = static_cast(img->data[0]); for (unsigned int y = 0; y < dim_y; y++) { unsigned int ymod = y_flip ? dim_y - y - 1 : y; float *dst = buf + y * dim_x * 4; for (unsigned int x = 0; x < dim_x; x++) { dst[4 * x ] = data32[(4 * dim_x * ymod) + (4 * x )]; dst[4 * x + 1] = data32[(4 * dim_x * ymod) + (4 * x + 1)]; dst[4 * x + 2] = data32[(4 * dim_x * ymod) + (4 * x + 2)]; dst[4 * x + 3] = data32[(4 * dim_x * ymod) + (4 * x + 3)]; } } } return buf; } /* See header for documentation. */ uint8_t* unorm8x4_array_from_astc_img( const astcenc_image* img, bool y_flip ) { unsigned int dim_x = img->dim_x; unsigned int dim_y = img->dim_y; uint8_t* buf = new uint8_t[4 * dim_x * dim_y]; if (img->data_type == ASTCENC_TYPE_U8) { uint8_t* data8 = static_cast(img->data[0]); for (unsigned int y = 0; y < dim_y; y++) { unsigned int ymod = y_flip ? dim_y - y - 1 : y; uint8_t* dst = buf + y * dim_x * 4; for (unsigned int x = 0; x < dim_x; x++) { dst[4 * x ] = data8[(4 * dim_x * ymod) + (4 * x )]; dst[4 * x + 1] = data8[(4 * dim_x * ymod) + (4 * x + 1)]; dst[4 * x + 2] = data8[(4 * dim_x * ymod) + (4 * x + 2)]; dst[4 * x + 3] = data8[(4 * dim_x * ymod) + (4 * x + 3)]; } } } else if (img->data_type == ASTCENC_TYPE_F16) { uint16_t* data16 = static_cast(img->data[0]); for (unsigned int y = 0; y < dim_y; y++) { unsigned int ymod = y_flip ? dim_y - y - 1 : y; uint8_t* dst = buf + y * dim_x * 4; for (unsigned int x = 0; x < dim_x; x++) { vint4 colori( data16[(4 * dim_x * ymod) + (4 * x )], data16[(4 * dim_x * ymod) + (4 * x + 1)], data16[(4 * dim_x * ymod) + (4 * x + 2)], data16[(4 * dim_x * ymod) + (4 * x + 3)] ); vfloat4 color = float16_to_float(colori); color = clamp(0.0f, 1.0f, color) * 255.0f; colori = float_to_int_rtn(color); pack_low_bytes(colori); store_nbytes(colori, dst + 4 * x); } } } else // if (img->data_type == ASTCENC_TYPE_F32) { assert(img->data_type == ASTCENC_TYPE_F32); float* data32 = static_cast(img->data[0]); for (unsigned int y = 0; y < dim_y; y++) { unsigned int ymod = y_flip ? dim_y - y - 1 : y; uint8_t* dst = buf + y * dim_x * 4; for (unsigned int x = 0; x < dim_x; x++) { dst[4 * x ] = static_cast(astc::flt2int_rtn(astc::clamp1f(data32[(4 * dim_x * ymod) + (4 * x )]) * 255.0f)); dst[4 * x + 1] = static_cast(astc::flt2int_rtn(astc::clamp1f(data32[(4 * dim_x * ymod) + (4 * x + 1)]) * 255.0f)); dst[4 * x + 2] = static_cast(astc::flt2int_rtn(astc::clamp1f(data32[(4 * dim_x * ymod) + (4 * x + 2)]) * 255.0f)); dst[4 * x + 3] = static_cast(astc::flt2int_rtn(astc::clamp1f(data32[(4 * dim_x * ymod) + (4 * x + 3)]) * 255.0f)); } } } return buf; }