/* * Copyright (C) 2017 The Android Open Source Project * * 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 "TexWrapper.h" #include "glError.h" #include "log/log.h" #include <fcntl.h> #include <malloc.h> #include <png.h> /* Create an new empty GL texture that will be filled later */ TexWrapper::TexWrapper() { GLuint textureId; glGenTextures(1, &textureId); if (textureId <= 0) { ALOGE("Didn't get a texture handle allocated: %s", getEGLError()); } else { // Store the basic texture properties id = textureId; w = 0; h = 0; } } /* Wrap a texture that already allocated. The wrapper takes ownership. */ TexWrapper::TexWrapper(GLuint textureId, unsigned width, unsigned height) { // Store the basic texture properties id = textureId; w = width; h = height; } TexWrapper::~TexWrapper() { // Give the texture ID back if (id > 0) { glDeleteTextures(1, &id); } id = -1; } /* Factory to build TexWrapper objects from a given PNG file */ TexWrapper* createTextureFromPng(const char * filename) { // Open the PNG file FILE *inputFile = fopen(filename, "rb"); if (inputFile == 0) { perror(filename); return nullptr; } // Read the file header and validate that it is a PNG static const int kSigSize = 8; png_byte header[kSigSize] = {0}; fread(header, 1, kSigSize, inputFile); if (png_sig_cmp(header, 0, kSigSize)) { printf("%s is not a PNG.\n", filename); fclose(inputFile); return nullptr; } // Set up our control structure png_structp pngControl = png_create_read_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL); if (!pngControl) { printf("png_create_read_struct failed.\n"); fclose(inputFile); return nullptr; } // Set up our image info structure png_infop pngInfo = png_create_info_struct(pngControl); if (!pngInfo) { printf("error: png_create_info_struct returned 0.\n"); png_destroy_read_struct(&pngControl, nullptr, nullptr); fclose(inputFile); return nullptr; } // Install an error handler if (setjmp(png_jmpbuf(pngControl))) { printf("libpng reported an error\n"); png_destroy_read_struct(&pngControl, &pngInfo, nullptr); fclose(inputFile); return nullptr; } // Set up the png reader and fetch the remaining bits of the header png_init_io(pngControl, inputFile); png_set_sig_bytes(pngControl, kSigSize); png_read_info(pngControl, pngInfo); // Get basic information about the PNG we're reading int bitDepth; int colorFormat; png_uint_32 width; png_uint_32 height; png_get_IHDR(pngControl, pngInfo, &width, &height, &bitDepth, &colorFormat, NULL, NULL, NULL); GLint format; switch(colorFormat) { case PNG_COLOR_TYPE_RGB: format = GL_RGB; break; case PNG_COLOR_TYPE_RGB_ALPHA: format = GL_RGBA; break; default: printf("%s: Unknown libpng color format %d.\n", filename, colorFormat); return nullptr; } // Refresh the values in the png info struct in case any transformation shave been applied. png_read_update_info(pngControl, pngInfo); int stride = png_get_rowbytes(pngControl, pngInfo); stride += 3 - ((stride-1) % 4); // glTexImage2d requires rows to be 4-byte aligned // Allocate storage for the pixel data png_byte * buffer = (png_byte*)malloc(stride * height); if (buffer == NULL) { printf("error: could not allocate memory for PNG image data\n"); png_destroy_read_struct(&pngControl, &pngInfo, nullptr); fclose(inputFile); return nullptr; } // libpng needs an array of pointers into the image data for each row png_byte ** rowPointers = (png_byte**)malloc(height * sizeof(png_byte*)); if (rowPointers == NULL) { printf("Failed to allocate temporary row pointers\n"); png_destroy_read_struct(&pngControl, &pngInfo, nullptr); free(buffer); fclose(inputFile); return nullptr; } for (unsigned int r = 0; r < height; r++) { rowPointers[r] = buffer + r*stride; } // Read in the actual image bytes png_read_image(pngControl, rowPointers); png_read_end(pngControl, nullptr); // Set up the OpenGL texture to contain this image GLuint textureId; glGenTextures(1, &textureId); glBindTexture(GL_TEXTURE_2D, textureId); // Send the image data to GL glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, width, height, 0, GL_RGBA, GL_UNSIGNED_BYTE, buffer); // Initialize the sampling properties (it seems the sample may not work if this isn't done) // The user of this texture may very well want to set their own filtering, but we're going // to pay the (minor) price of setting this up for them to avoid the dreaded "black image" if // they forget. glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST); // clean up png_destroy_read_struct(&pngControl, &pngInfo, nullptr); free(buffer); free(rowPointers); fclose(inputFile); glBindTexture(GL_TEXTURE_2D, 0); // Return the texture return new TexWrapper(textureId, width, height); }