// Copyright 2020 The SwiftShader Authors. All Rights Reserved. // // 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. #include "ASTC_Decoder.hpp" #include "System/Math.hpp" #ifdef SWIFTSHADER_ENABLE_ASTC # include "astc_codec_internals.h" #endif #include #include namespace { #ifdef SWIFTSHADER_ENABLE_ASTC void write_imageblock(unsigned char *img, // picture-block to initialize with image data. We assume that orig_data is valid const imageblock *pb, // output dimensions int xsize, int ysize, int zsize, // output format int bytes, int destPitchB, int destSliceB, bool isUnsignedByte, // block dimensions int xdim, int ydim, int zdim, // position to write the block to int xpos, int ypos, int zpos) { const float *fptr = pb->orig_data; const uint8_t *nptr = pb->nan_texel; for(int z = 0; z < zdim; z++) { for(int y = 0; y < ydim; y++) { for(int x = 0; x < xdim; x++) { int xi = xpos + x; int yi = ypos + y; int zi = zpos + z; if(xi >= 0 && yi >= 0 && zi >= 0 && xi < xsize && yi < ysize && zi < zsize) { unsigned char *pix = &img[zi * destSliceB + yi * destPitchB + xi * bytes]; if(isUnsignedByte) { if(*nptr) { // NaN-pixel, but we can't display it. Display purple instead. pix[0] = 0xFF; pix[1] = 0x00; pix[2] = 0xFF; pix[3] = 0xFF; } else { pix[0] = static_cast(sw::clamp(fptr[0], 0.0f, 1.0f) * 255.0f + 0.5f); pix[1] = static_cast(sw::clamp(fptr[1], 0.0f, 1.0f) * 255.0f + 0.5f); pix[2] = static_cast(sw::clamp(fptr[2], 0.0f, 1.0f) * 255.0f + 0.5f); pix[3] = static_cast(sw::clamp(fptr[3], 0.0f, 1.0f) * 255.0f + 0.5f); } } else { if(*nptr) { unsigned int *pixu = reinterpret_cast(pix); pixu[0] = pixu[1] = pixu[2] = pixu[3] = 0x7FFFFFFF; // QNaN } else { float *pixf = reinterpret_cast(pix); pixf[0] = fptr[0]; pixf[1] = fptr[1]; pixf[2] = fptr[2]; pixf[3] = fptr[3]; } } } fptr += 4; nptr++; } } } } #endif } // namespace void ASTC_Decoder::Decode(const unsigned char *source, unsigned char *dest, int destWidth, int destHeight, int destDepth, int bytes, int destPitchB, int destSliceB, int xBlockSize, int yBlockSize, int zBlockSize, int xblocks, int yblocks, int zblocks, bool isUnsignedByte) { #ifdef SWIFTSHADER_ENABLE_ASTC build_quantization_mode_table(); astc_decode_mode decode_mode = isUnsignedByte ? DECODE_LDR : DECODE_HDR; std::unique_ptr bsd(new block_size_descriptor); init_block_size_descriptor(xBlockSize, yBlockSize, zBlockSize, bsd.get()); std::unique_ptr ib(new imageblock); std::unique_ptr scb(new symbolic_compressed_block); for(int z = 0; z < zblocks; z++) { for(int y = 0; y < yblocks; y++) { for(int x = 0; x < xblocks; x++, source += 16) { physical_to_symbolic(bsd.get(), *(physical_compressed_block *)source, scb.get()); decompress_symbolic_block(decode_mode, bsd.get(), x * xBlockSize, y * yBlockSize, z * zBlockSize, scb.get(), ib.get()); write_imageblock(dest, ib.get(), destWidth, destHeight, destDepth, bytes, destPitchB, destSliceB, isUnsignedByte, xBlockSize, yBlockSize, zBlockSize, x * xBlockSize, y * yBlockSize, z * zBlockSize); } } } term_block_size_descriptor(bsd.get()); #endif }