mirror of
https://github.com/yuzu-emu/FasTC.git
synced 2025-01-08 22:05:37 +00:00
Add initial png writing routines.
This commit is contained in:
parent
545a6f68e2
commit
0dbf5a08cc
|
@ -1,11 +1,13 @@
|
||||||
|
|
||||||
SET( SOURCES
|
SET( SOURCES
|
||||||
"src/ImageFile.cpp"
|
"src/ImageWriter.cpp"
|
||||||
"src/ImageLoader.cpp"
|
"src/ImageLoader.cpp"
|
||||||
"src/FileStream.cpp"
|
"src/FileStream.cpp"
|
||||||
|
"src/ImageFile.cpp"
|
||||||
)
|
)
|
||||||
|
|
||||||
SET( HEADERS
|
SET( HEADERS
|
||||||
|
"config/ImageWriter.h.in"
|
||||||
"config/ImageLoader.h.in"
|
"config/ImageLoader.h.in"
|
||||||
"include/ImageFile.h"
|
"include/ImageFile.h"
|
||||||
"include/FileStream.h"
|
"include/FileStream.h"
|
||||||
|
@ -17,6 +19,8 @@ IF( PNG_FOUND )
|
||||||
|
|
||||||
SET( SOURCES ${SOURCES} "src/ImageLoaderPNG.cpp" )
|
SET( SOURCES ${SOURCES} "src/ImageLoaderPNG.cpp" )
|
||||||
SET( HEADERS ${HEADERS} "src/ImageLoaderPNG.h" )
|
SET( HEADERS ${HEADERS} "src/ImageLoaderPNG.h" )
|
||||||
|
SET( SOURCES ${SOURCES} "src/ImageWriterPNG.cpp" )
|
||||||
|
SET( HEADERS ${HEADERS} "src/ImageWriterPNG.h" )
|
||||||
ENDIF()
|
ENDIF()
|
||||||
|
|
||||||
CONFIGURE_FILE(
|
CONFIGURE_FILE(
|
||||||
|
@ -24,6 +28,11 @@ CONFIGURE_FILE(
|
||||||
"include/ImageLoader.h"
|
"include/ImageLoader.h"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
CONFIGURE_FILE(
|
||||||
|
"config/ImageWriter.h.in"
|
||||||
|
"include/ImageWriter.h"
|
||||||
|
)
|
||||||
|
|
||||||
INCLUDE_DIRECTORIES( ${TexC_BINARY_DIR}/IO/include )
|
INCLUDE_DIRECTORIES( ${TexC_BINARY_DIR}/IO/include )
|
||||||
INCLUDE_DIRECTORIES( ${TexC_SOURCE_DIR}/IO/include )
|
INCLUDE_DIRECTORIES( ${TexC_SOURCE_DIR}/IO/include )
|
||||||
INCLUDE_DIRECTORIES( ${TexC_SOURCE_DIR}/Core/include )
|
INCLUDE_DIRECTORIES( ${TexC_SOURCE_DIR}/Core/include )
|
||||||
|
@ -35,5 +44,5 @@ ADD_LIBRARY(TexCompIO
|
||||||
|
|
||||||
IF( PNG_FOUND )
|
IF( PNG_FOUND )
|
||||||
TARGET_LINK_LIBRARIES( TexCompIO ${PNG_LIBRARY} )
|
TARGET_LINK_LIBRARIES( TexCompIO ${PNG_LIBRARY} )
|
||||||
TARGET_LINK_LIBRARIES( TexCompIO ${ZLIB_LIBRARY} )
|
TARGET_LINK_LIBRARIES( TexCompIO ${ZLIB_LIBRARY} )
|
||||||
ENDIF()
|
ENDIF()
|
||||||
|
|
45
IO/config/ImageWriter.h.in
Normal file
45
IO/config/ImageWriter.h.in
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
#ifndef _IMAGE_WRITER_H_
|
||||||
|
#define _IMAGE_WRITER_H_
|
||||||
|
|
||||||
|
#include "ImageFileFormat.h"
|
||||||
|
#include "TexCompTypes.h"
|
||||||
|
|
||||||
|
class ImageWriter {
|
||||||
|
|
||||||
|
protected:
|
||||||
|
|
||||||
|
uint32 m_Width;
|
||||||
|
uint32 m_Height;
|
||||||
|
|
||||||
|
const uint8 *m_PixelData;
|
||||||
|
uint32 m_RawFileDataSz;
|
||||||
|
uint8 *m_RawFileData;
|
||||||
|
|
||||||
|
ImageWriter(const int width, const int height, const uint8 *rawData)
|
||||||
|
: m_PixelData(rawData)
|
||||||
|
, m_Width(width), m_Height(height)
|
||||||
|
, m_RawFileDataSz(256)
|
||||||
|
, m_RawFileData(new uint8[m_RawFileDataSz])
|
||||||
|
{ }
|
||||||
|
|
||||||
|
uint32 GetChannelForPixel(uint32 x, uint32 y, uint32 ch);
|
||||||
|
|
||||||
|
public:
|
||||||
|
virtual ~ImageWriter() {
|
||||||
|
if(m_RawFileData) {
|
||||||
|
delete m_RawFileData;
|
||||||
|
m_RawFileData = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
uint32 GetWidth() const { return m_Width; }
|
||||||
|
uint32 GetHeight() const { return m_Height; }
|
||||||
|
uint32 GetImageDataSz() const { return m_Width * m_Height * 4; }
|
||||||
|
uint32 GetRawImageDataSz() const { return m_RawFileDataSz; }
|
||||||
|
uint8 *GetRawImageData() const { return m_RawFileData; }
|
||||||
|
virtual bool WriteImage() = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
#cmakedefine PNG_FOUND
|
||||||
|
|
||||||
|
#endif // _IMAGE_LOADER_H_
|
7
IO/src/ImageWriter.cpp
Normal file
7
IO/src/ImageWriter.cpp
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
#include "ImageWriter.h"
|
||||||
|
|
||||||
|
uint32 ImageWriter::GetChannelForPixel(uint32 x, uint32 y, uint32 ch) {
|
||||||
|
uint32 bytesPerRow = GetWidth() * 4;
|
||||||
|
uint32 byteLocation = y * bytesPerRow + x*4 + ch;
|
||||||
|
return m_PixelData[byteLocation];
|
||||||
|
}
|
110
IO/src/ImageWriterPNG.cpp
Normal file
110
IO/src/ImageWriterPNG.cpp
Normal file
|
@ -0,0 +1,110 @@
|
||||||
|
#include "ImageWriterPNG.h"
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include "Image.h"
|
||||||
|
|
||||||
|
#include <png.h>
|
||||||
|
|
||||||
|
class PNGStreamWriter {
|
||||||
|
public:
|
||||||
|
static void WriteDataToStream(
|
||||||
|
png_structp png_ptr,
|
||||||
|
png_bytep outBytes,
|
||||||
|
png_size_t byteCountToWrite
|
||||||
|
) {
|
||||||
|
png_voidp io_ptr = png_get_io_ptr( png_ptr );
|
||||||
|
if( io_ptr == NULL ) {
|
||||||
|
fprintf(stderr, "Write callback had invalid io pointer.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
ImageWriterPNG &writer = *(ImageWriterPNG *)(io_ptr);
|
||||||
|
|
||||||
|
while(writer.m_StreamPosition + byteCountToWrite > writer.m_RawFileDataSz) {
|
||||||
|
uint8 *newData = new uint8[writer.m_RawFileDataSz << 1];
|
||||||
|
memcpy(newData, writer.m_RawFileData, writer.m_RawFileDataSz);
|
||||||
|
writer.m_RawFileDataSz <<= 1;
|
||||||
|
delete writer.m_RawFileData;
|
||||||
|
writer.m_RawFileData = newData;
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned char *stream = &(writer.m_RawFileData[writer.m_StreamPosition]);
|
||||||
|
memcpy(stream, outBytes, byteCountToWrite);
|
||||||
|
|
||||||
|
writer.m_StreamPosition += byteCountToWrite;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void FlushStream(png_structp png_ptr) { /* Do nothing... */ }
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
ImageWriterPNG::ImageWriterPNG(const Image &im)
|
||||||
|
: ImageWriter(im.GetWidth(), im.GetHeight(), im.RawData())
|
||||||
|
, m_StreamPosition(0)
|
||||||
|
, m_TotalBytesWritten(0)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ImageWriterPNG::WriteImage() {
|
||||||
|
|
||||||
|
png_structp png_ptr = NULL;
|
||||||
|
png_infop info_ptr = NULL;
|
||||||
|
png_byte ** row_pointers = NULL;
|
||||||
|
int pixel_size = 4;
|
||||||
|
int depth = 8;
|
||||||
|
|
||||||
|
png_ptr = png_create_write_struct (PNG_LIBPNG_VER_STRING, NULL, NULL, NULL);
|
||||||
|
if (png_ptr == NULL) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
info_ptr = png_create_info_struct (png_ptr);
|
||||||
|
if (info_ptr == NULL) {
|
||||||
|
png_destroy_write_struct (&png_ptr, &info_ptr);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Set image attributes. */
|
||||||
|
|
||||||
|
png_set_IHDR (png_ptr,
|
||||||
|
info_ptr,
|
||||||
|
m_Width,
|
||||||
|
m_Height,
|
||||||
|
depth,
|
||||||
|
PNG_COLOR_TYPE_RGBA,
|
||||||
|
PNG_INTERLACE_NONE,
|
||||||
|
PNG_COMPRESSION_TYPE_DEFAULT,
|
||||||
|
PNG_FILTER_TYPE_DEFAULT);
|
||||||
|
|
||||||
|
/* Initialize rows of PNG. */
|
||||||
|
|
||||||
|
row_pointers = (png_byte **)png_malloc (png_ptr, m_Height * sizeof (png_byte *));
|
||||||
|
for (int y = 0; y < m_Height; ++y) {
|
||||||
|
png_byte *row = (png_byte *)png_malloc (png_ptr, sizeof (uint8) * m_Width * pixel_size);
|
||||||
|
|
||||||
|
row_pointers[y] = row;
|
||||||
|
|
||||||
|
for (int x = 0; x < m_Width; ++x) {
|
||||||
|
for(int ch = 0; ch < 4; ch++) {
|
||||||
|
*row++ = GetChannelForPixel(x, y, ch);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
png_set_write_fn(png_ptr, this, PNGStreamWriter::WriteDataToStream, PNGStreamWriter::FlushStream);
|
||||||
|
png_set_rows (png_ptr, info_ptr, row_pointers);
|
||||||
|
png_write_png (png_ptr, info_ptr, PNG_TRANSFORM_IDENTITY, NULL);
|
||||||
|
|
||||||
|
for (int y = 0; y < m_Height; y++) {
|
||||||
|
png_free (png_ptr, row_pointers[y]);
|
||||||
|
}
|
||||||
|
png_free (png_ptr, row_pointers);
|
||||||
|
|
||||||
|
png_destroy_write_struct (&png_ptr, &info_ptr);
|
||||||
|
|
||||||
|
m_RawFileDataSz = m_StreamPosition;
|
||||||
|
return true;
|
||||||
|
}
|
20
IO/src/ImageWriterPNG.h
Normal file
20
IO/src/ImageWriterPNG.h
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
#ifndef _IMAGE_WRITER_PNG_H_
|
||||||
|
#define _IMAGE_WRITER_PNG_H_
|
||||||
|
|
||||||
|
#include "ImageWriter.h"
|
||||||
|
|
||||||
|
// Forward Declare
|
||||||
|
class Image;
|
||||||
|
class ImageWriterPNG : public ImageWriter {
|
||||||
|
public:
|
||||||
|
ImageWriterPNG(const Image &);
|
||||||
|
virtual ~ImageWriterPNG();
|
||||||
|
|
||||||
|
virtual bool WriteImage();
|
||||||
|
private:
|
||||||
|
uint32 m_StreamPosition;
|
||||||
|
uint32 m_TotalBytesWritten;
|
||||||
|
friend class PNGStreamWriter;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // _IMAGE_LOADER_H_
|
Loading…
Reference in a new issue