OK, I originally intended to fix FrameBuffer::scanline for SDL2,

which I though a just a one or two-liner equivalent to glReadPixels.
Turns out that SDL2 doesn't return 24-bit RGB data (only 16 or 32-bit),
so the PNGLibrary had to massage the data.  In the process, converted
the PNG saving functions to actually use PNG library functions, which
greatly simplified converting packing from 32 -> 24-bit.  Historically,
the PNG save functions were written before libpng was a requirement,
so they were hand-rolled.  Once libpng was integrated into the codebase,
it didn't make much sense to not use it anymore.

So an expected two-line code fix became this commit ...


git-svn-id: svn://svn.code.sf.net/p/stella/code/trunk@2919 8b62c5a3-ac7e-4cc8-8f21-d9a121418aba
This commit is contained in:
stephena 2014-06-13 13:35:41 +00:00
parent a0aa1c8e0a
commit d275773413
10 changed files with 245 additions and 235 deletions

View File

@ -234,15 +234,12 @@ FBSurface* FrameBufferSDL2::createSurface(uInt32 w, uInt32 h,
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void FrameBufferSDL2::scanline(uInt32 row, uInt8* data) const void FrameBufferSDL2::readPixels(const GUI::Rect& rect,
uInt8* pixels, uInt32 pitch) const
{ {
#if 0 //FIXSDL SDL_Rect r;
// Invert the row, since OpenGL rows start at the bottom r.x = rect.x(); r.y = rect.y();
// of the framebuffer r.w = rect.width(); r.h = rect.height();
const GUI::Rect& image = imageRect();
row = image.height() + image.y() - row - 1;
p_gl.PixelStorei(GL_PACK_ALIGNMENT, 1); SDL_RenderReadPixels(myRenderer, &r, SDL_PIXELFORMAT_ARGB8888, pixels, pitch);
p_gl.ReadPixels(image.x(), row, image.width(), 1, GL_RGB, GL_UNSIGNED_BYTE, data);
#endif
} }

View File

@ -96,12 +96,16 @@ class FrameBufferSDL2 : public FrameBuffer
bool isDoubleBuffered() const { return myDblBufferedFlag; } bool isDoubleBuffered() const { return myDblBufferedFlag; }
/** /**
This method is called to get the specified scanline data. This method is called to get the specified ARGB data from the viewable
FrameBuffer area. Note that this isn't the same as any internal
surfaces that may be in use; it should return the actual data as it
is currently seen onscreen.
@param row The row we are looking for @param rect The bounding rectangle for the buffer
@param data The actual pixel data (in bytes) @param buffer The actual pixel data in ARGB8888 format
@param pitch The pitch (in bytes) for the pixel data
*/ */
void scanline(uInt32 row, uInt8* data) const; void readPixels(const GUI::Rect& rect, uInt8* buffer, uInt32 pitch) const;
protected: protected:
////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////

View File

@ -17,7 +17,6 @@
// $Id$ // $Id$
//============================================================================ //============================================================================
#include <zlib.h>
#include <fstream> #include <fstream>
#include <cstring> #include <cstring>
#include <sstream> #include <sstream>
@ -31,7 +30,8 @@
#include "PNGLibrary.hxx" #include "PNGLibrary.hxx"
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
PNGLibrary::PNGLibrary() PNGLibrary::PNGLibrary(const FrameBuffer& fb)
: myFB(fb)
{ {
} }
@ -44,10 +44,9 @@ PNGLibrary::~PNGLibrary()
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
bool PNGLibrary::loadImage(const string& filename, void PNGLibrary::loadImage(const string& filename, FBSurface& surface)
const FrameBuffer& fb, FBSurface& surface)
{ {
#define readImageERROR(s) { err_message = s; goto done; } #define loadImageERROR(s) { err_message = s; goto done; }
png_structp png_ptr = NULL; png_structp png_ptr = NULL;
png_infop info_ptr = NULL; png_infop info_ptr = NULL;
@ -57,18 +56,18 @@ bool PNGLibrary::loadImage(const string& filename,
ifstream in(filename.c_str(), ios_base::binary); ifstream in(filename.c_str(), ios_base::binary);
if(!in.is_open()) if(!in.is_open())
readImageERROR("No image found"); loadImageERROR("No image found");
// Create the PNG loading context structure // Create the PNG loading context structure
png_ptr = png_create_read_struct(PNG_LIBPNG_VER_STRING, NULL, png_ptr = png_create_read_struct(PNG_LIBPNG_VER_STRING, NULL,
png_user_error, png_user_warn); png_user_error, png_user_warn);
if(png_ptr == NULL) if(png_ptr == NULL)
readImageERROR("Couldn't allocate memory for PNG file"); loadImageERROR("Couldn't allocate memory for PNG file");
// Allocate/initialize the memory for image information. REQUIRED. // Allocate/initialize the memory for image information. REQUIRED.
info_ptr = png_create_info_struct(png_ptr); info_ptr = png_create_info_struct(png_ptr);
if(info_ptr == NULL) if(info_ptr == NULL)
readImageERROR("Couldn't create image information for PNG file"); loadImageERROR("Couldn't create image information for PNG file");
// Set up the input control // Set up the input control
png_set_read_fn(png_ptr, &in, png_read_data); png_set_read_fn(png_ptr, &in, png_read_data);
@ -92,20 +91,20 @@ bool PNGLibrary::loadImage(const string& filename,
} }
else if(color_type == PNG_COLOR_TYPE_GRAY || color_type == PNG_COLOR_TYPE_GRAY_ALPHA) else if(color_type == PNG_COLOR_TYPE_GRAY || color_type == PNG_COLOR_TYPE_GRAY_ALPHA)
{ {
readImageERROR("Greyscale PNG images not supported"); loadImageERROR("Greyscale PNG images not supported");
} }
else if(color_type == PNG_COLOR_TYPE_PALETTE) else if(color_type == PNG_COLOR_TYPE_PALETTE)
{ {
readImageERROR("Paletted PNG images not supported"); png_set_palette_to_rgb(png_ptr);
} }
else if(color_type != PNG_COLOR_TYPE_RGB) else if(color_type != PNG_COLOR_TYPE_RGB)
{ {
readImageERROR("Unknown format in PNG image"); loadImageERROR("Unknown format in PNG image");
} }
// Create/initialize storage area for the current image // Create/initialize storage area for the current image
if(!allocateStorage(iwidth, iheight)) if(!allocateStorage(iwidth, iheight))
readImageERROR("Not enough memory to read PNG file"); loadImageERROR("Not enough memory to read PNG file");
// The PNG read function expects an array of rows, not a single 1-D array // The PNG read function expects an array of rows, not a single 1-D array
for(uInt32 irow = 0, offset = 0; irow < ReadInfo.height; ++irow, offset += ReadInfo.pitch) for(uInt32 irow = 0, offset = 0; irow < ReadInfo.height; ++irow, offset += ReadInfo.pitch)
@ -118,7 +117,7 @@ bool PNGLibrary::loadImage(const string& filename,
png_read_end(png_ptr, info_ptr); png_read_end(png_ptr, info_ptr);
// Scale image to surface dimensions // Scale image to surface dimensions
scaleImagetoSurface(fb, surface); scaleImagetoSurface(surface);
// Cleanup // Cleanup
done: done:
@ -127,142 +126,161 @@ done:
if(err_message) if(err_message)
throw err_message; throw err_message;
else
return true;
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
string PNGLibrary::saveImage(const string& filename, void PNGLibrary::saveImage(const string& filename, const Properties& props)
const FrameBuffer& framebuffer,
const Properties& props)
{ {
#define saveImageERROR(s) { err_message = s; goto done; }
png_structp png_ptr = NULL;
png_infop info_ptr = NULL;
const char* err_message = NULL;
const GUI::Rect& imageR = myFB.imageRect();
png_uint_32 width = imageR.width(), height = imageR.height();
png_bytep image = new png_byte[width * height * 4];
png_bytep row_pointers[height];
ofstream out(filename.c_str(), ios_base::binary); ofstream out(filename.c_str(), ios_base::binary);
if(!out.is_open()) if(!out.is_open())
return "ERROR: Couldn't create snapshot file"; saveImageERROR("ERROR: Couldn't create snapshot file");
// Get actual image dimensions. which are not always the same // Create the PNG saving context structure
// as the framebuffer dimensions png_ptr = png_create_write_struct(PNG_LIBPNG_VER_STRING, NULL,
const GUI::Rect& image = framebuffer.imageRect(); png_user_error, png_user_warn);
uInt32 width = image.width(), height = image.height(), if(png_ptr == NULL)
pitch = width * 3; saveImageERROR("Couldn't allocate memory for PNG file");
uInt8* buffer = new uInt8[(pitch + 1) * height];
// Fill the buffer with scanline data // Allocate/initialize the memory for image information. REQUIRED.
uInt8* buf_ptr = buffer; info_ptr = png_create_info_struct(png_ptr);
for(uInt32 row = 0; row < height; row++) if(info_ptr == NULL)
{ saveImageERROR("Couldn't create image information for PNG file");
*buf_ptr++ = 0; // first byte of row is filter type
framebuffer.scanline(row, buf_ptr); // get another scanline
buf_ptr += pitch; // add pitch
}
return saveBufferToPNG(out, buffer, width, height, props, // Set up the output control
framebuffer.tiaSurface().effectsInfo()); png_set_write_fn(png_ptr, &out, png_write_data, png_io_flush);
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // Write PNG header info
string PNGLibrary::saveImage(const string& filename, png_set_IHDR(png_ptr, info_ptr, width, height, 8,
const FrameBuffer& framebuffer, const TIA& tia, PNG_COLOR_TYPE_RGB, PNG_INTERLACE_NONE, PNG_COMPRESSION_TYPE_DEFAULT,
const Properties& props) PNG_FILTER_TYPE_DEFAULT);
{
ofstream out(filename.c_str(), ios_base::binary);
if(!out.is_open())
return "ERROR: Couldn't create snapshot file";
uInt32 width = tia.width(), height = tia.height(); // Write comments
uInt8* buffer = new uInt8[(width*3*2 + 1) * height]; writeComments(png_ptr, info_ptr, props);
// Fill the buffer with pixels from the mediasrc // Write the file header information. REQUIRED
uInt8 r, g, b; png_write_info(png_ptr, info_ptr);
uInt8* buf_ptr = buffer;
for(uInt32 y = 0; y < height; ++y)
{
*buf_ptr++ = 0; // first byte of row is filter type
for(uInt32 x = 0; x < width; ++x)
{
uInt32 pixel = framebuffer.tiaSurface().pixel(y*width+x);
framebuffer.getRGB(pixel, &r, &g, &b);
*buf_ptr++ = r;
*buf_ptr++ = g;
*buf_ptr++ = b;
*buf_ptr++ = r;
*buf_ptr++ = g;
*buf_ptr++ = b;
}
}
return saveBufferToPNG(out, buffer, width << 1, height, props, // Pack pixels into bytes
framebuffer.tiaSurface().effectsInfo()); png_set_packing(png_ptr);
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // Swap location of alpha bytes from ARGB to RGBA
string PNGLibrary::saveBufferToPNG(ofstream& out, uInt8* buffer, png_set_swap_alpha(png_ptr);
uInt32 width, uInt32 height,
const Properties& props,
const string& effectsInfo)
{
uInt8* compmem = (uInt8*) NULL;
try // Pack ARGB into RGB
{ png_set_filler(png_ptr, 0, PNG_FILLER_AFTER);
// PNG file header
uInt8 header[8] = { 137, 80, 78, 71, 13, 10, 26, 10 };
out.write((const char*)header, 8);
// PNG IHDR // Flip BGR pixels to RGB
uInt8 ihdr[13]; png_set_bgr(png_ptr);
ihdr[0] = width >> 24; // width
ihdr[1] = width >> 16;
ihdr[2] = width >> 8;
ihdr[3] = width & 0xFF;
ihdr[4] = height >> 24; // height
ihdr[5] = height >> 16;
ihdr[6] = height >> 8;
ihdr[7] = height & 0xFF;
ihdr[8] = 8; // 8 bits per sample (24 bits per pixel)
ihdr[9] = 2; // PNG_COLOR_TYPE_RGB
ihdr[10] = 0; // PNG_COMPRESSION_TYPE_DEFAULT
ihdr[11] = 0; // PNG_FILTER_TYPE_DEFAULT
ihdr[12] = 0; // PNG_INTERLACE_NONE
writePNGChunk(out, "IHDR", ihdr, 13);
// Compress the data with zlib // Get framebuffer surface pixel data (we get ABGR format)
uLongf compmemsize = (uLongf)((height * (width + 1) * 3 * 1.001 + 1) + 12); myFB.readPixels(imageR, image, width*4);
compmem = new uInt8[compmemsize];
if(compmem == NULL ||
(compress(compmem, &compmemsize, buffer, height * (width * 3 + 1)) != Z_OK))
throw "ERROR: Couldn't compress PNG";
// Write the compressed framebuffer data // Set up pointers into "image" byte array
writePNGChunk(out, "IDAT", compmem, compmemsize); for(png_uint_32 k = 0; k < height; ++k)
row_pointers[k] = image + k*width*4;
// Add some info about this snapshot // Write the entire image in one go
ostringstream text; png_write_image(png_ptr, row_pointers);
text << "Stella " << STELLA_VERSION << " (Build " << STELLA_BUILD << ") ["
<< BSPF_ARCH << "]";
writePNGText(out, "Software", text.str()); // We're finished writing
writePNGText(out, "ROM Name", props.get(Cartridge_Name)); png_write_end(png_ptr, info_ptr);
writePNGText(out, "ROM MD5", props.get(Cartridge_MD5));
writePNGText(out, "TV Effects", effectsInfo);
// Finish up
writePNGChunk(out, "IEND", 0, 0);
// Cleanup // Cleanup
if(buffer) delete[] buffer; done:
if(compmem) delete[] compmem; if(png_ptr)
out.close(); png_destroy_write_struct(&png_ptr, &info_ptr);
if(image)
delete[] image;
if(err_message)
throw err_message;
}
return "Snapshot saved"; // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
} void PNGLibrary::saveImage(const string& filename, const TIA& tia,
catch(const char* msg) const Properties& props)
{ {
if(buffer) delete[] buffer; #define saveImageERROR(s) { err_message = s; goto done; }
if(compmem) delete[] compmem;
out.close(); png_structp png_ptr = NULL;
return msg; png_infop info_ptr = NULL;
const char* err_message = NULL;
png_uint_32 tiaw = tia.width(), width = tiaw*2, height = tia.height();
png_bytep image = new png_byte[width * height * 3];
png_bytep row_pointers[height];
uInt8 r, g, b;
uInt8* buf_ptr = image;
ofstream out(filename.c_str(), ios_base::binary);
if(!out.is_open())
saveImageERROR("ERROR: Couldn't create snapshot file");
// Create the PNG saving context structure
png_ptr = png_create_write_struct(PNG_LIBPNG_VER_STRING, NULL,
png_user_error, png_user_warn);
if(png_ptr == NULL)
saveImageERROR("Couldn't allocate memory for PNG file");
// Allocate/initialize the memory for image information. REQUIRED.
info_ptr = png_create_info_struct(png_ptr);
if(info_ptr == NULL)
saveImageERROR("Couldn't create image information for PNG file");
// Set up the output control
png_set_write_fn(png_ptr, &out, png_write_data, png_io_flush);
// Write PNG header info
png_set_IHDR(png_ptr, info_ptr, width, height, 8,
PNG_COLOR_TYPE_RGB, PNG_INTERLACE_NONE, PNG_COMPRESSION_TYPE_DEFAULT,
PNG_FILTER_TYPE_DEFAULT);
// Write comments
writeComments(png_ptr, info_ptr, props);
// Write the file header information. REQUIRED
png_write_info(png_ptr, info_ptr);
// Fill the buffer with pixels from the tia, scaled 2x horizontally
for(uInt32 y = 0; y < height; ++y)
{
for(uInt32 x = 0; x < tiaw; ++x)
{
uInt32 pixel = myFB.tiaSurface().pixel(y*tiaw+x);
myFB.getRGB(pixel, &r, &g, &b);
*buf_ptr++ = r;
*buf_ptr++ = g;
*buf_ptr++ = b;
*buf_ptr++ = r;
*buf_ptr++ = g;
*buf_ptr++ = b;
} }
// Set up pointers into "image" byte array
buf_ptr = row_pointers[y] = image + y*width*3;
}
// Write the entire image in one go
png_write_image(png_ptr, row_pointers);
// We're finished writing
png_write_end(png_ptr, info_ptr);
// Cleanup
done:
if(png_ptr)
png_destroy_write_struct(&png_ptr, &info_ptr);
if(image)
delete[] image;
if(err_message)
throw err_message;
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
@ -308,7 +326,7 @@ bool PNGLibrary::allocateStorage(png_uint_32 w, png_uint_32 h)
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void PNGLibrary::scaleImagetoSurface(const FrameBuffer& fb, FBSurface& surface) void PNGLibrary::scaleImagetoSurface(FBSurface& surface)
{ {
// Figure out the original zoom level of the snapshot // Figure out the original zoom level of the snapshot
// All snapshots generated by Stella are at most some multiple of 320 // All snapshots generated by Stella are at most some multiple of 320
@ -345,7 +363,7 @@ void PNGLibrary::scaleImagetoSurface(const FrameBuffer& fb, FBSurface& surface)
uInt32* l_ptr = ReadInfo.line; uInt32* l_ptr = ReadInfo.line;
for(uInt32 icol = 0; icol < ReadInfo.width; icol += izoom, i_ptr += i_offset) for(uInt32 icol = 0; icol < ReadInfo.width; icol += izoom, i_ptr += i_offset)
{ {
uInt32 pixel = fb.mapRGB(*i_ptr, *(i_ptr+1), *(i_ptr+2)); uInt32 pixel = myFB.mapRGB(*i_ptr, *(i_ptr+1), *(i_ptr+2));
uInt32 xstride = szoom; uInt32 xstride = szoom;
while(xstride--) while(xstride--)
*l_ptr++ = pixel; *l_ptr++ = pixel;
@ -359,50 +377,30 @@ void PNGLibrary::scaleImagetoSurface(const FrameBuffer& fb, FBSurface& surface)
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void PNGLibrary::writePNGChunk(ofstream& out, const char* type, void PNGLibrary::writeComments(png_structp png_ptr, png_infop info_ptr,
uInt8* data, int size) const Properties& props)
{ {
// Stuff the length/type into the buffer // Pre-processor voodoo to make the code shorter
uInt8 temp[8]; #define CONVERT_TO_PNGTEXT(_idx, _key, _text) \
temp[0] = size >> 24; char key##_idx[] = _key; \
temp[1] = size >> 16; char text##_idx[256]; \
temp[2] = size >> 8; strncpy(text##_idx, _text.c_str(), 255); \
temp[3] = size; text_ptr[_idx].key = key##_idx; \
temp[4] = type[0]; text_ptr[_idx].text = text##_idx; \
temp[5] = type[1]; text_ptr[_idx].compression = PNG_TEXT_COMPRESSION_NONE; \
temp[6] = type[2]; text_ptr[_idx].text_length = 0;
temp[7] = type[3];
// Write the header ostringstream version;
out.write((const char*)temp, 8); version << "Stella " << STELLA_VERSION << " (Build " << STELLA_BUILD << ") ["
<< BSPF_ARCH << "]";
// Append the actual data png_text text_ptr[4];
uInt32 crc = crc32(0, temp + 4, 4); CONVERT_TO_PNGTEXT(0, "Software", version.str());
if(size > 0) CONVERT_TO_PNGTEXT(1, "ROM Name", props.get(Cartridge_Name));
{ CONVERT_TO_PNGTEXT(2, "ROM MD5", props.get(Cartridge_MD5));
out.write((const char*)data, size); CONVERT_TO_PNGTEXT(3, "TV Effects", myFB.tiaSurface().effectsInfo());
crc = crc32(crc, data, size);
}
// Write the CRC png_set_text(png_ptr, info_ptr, text_ptr, 4);
temp[0] = crc >> 24;
temp[1] = crc >> 16;
temp[2] = crc >> 8;
temp[3] = crc;
out.write((const char*)temp, 4);
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void PNGLibrary::writePNGText(ofstream& out, const string& key, const string& text)
{
int length = key.length() + 1 + text.length() + 1;
uInt8* data = new uInt8[length];
strcpy((char*)data, key.c_str());
strcpy((char*)data + key.length() + 1, text.c_str());
writePNGChunk(out, "tEXt", data, length-1);
delete[] data;
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -

View File

@ -40,7 +40,7 @@ class TIA;
class PNGLibrary class PNGLibrary
{ {
public: public:
PNGLibrary(); PNGLibrary(const FrameBuffer& fb);
virtual ~PNGLibrary(); virtual ~PNGLibrary();
/** /**
@ -48,39 +48,44 @@ class PNGLibrary
scaling the image to the surface bounds. scaling the image to the surface bounds.
@param filename The filename to load the PNG image @param filename The filename to load the PNG image
@param fb The main framebuffer of the application
@param surface The FBSurface into which to place the PNG data @param surface The FBSurface into which to place the PNG data
@return On success, the FBSurface containing image data and a @return On success, the FBSurface containing image data, otherwise a
result of true, otherwise a const char* exception is thrown const char* exception is thrown containing a more detailed
containing a more detailed error message. error message.
*/ */
bool loadImage(const string& filename, const FrameBuffer& fb, FBSurface& surface); void loadImage(const string& filename, FBSurface& surface);
/** /**
Save the current TIA image to a PNG file using data from the Framebuffer. Save the current FrameBuffer image to a PNG file. Note that in most cases
Any postprocessing/filtering will be included. this will be a TIA image, but it could actually be used for *any* mode.
@param filename The filename to save the PNG image @param filename The filename to save the PNG image
@param framebuffer The framebuffer containing the image data
@param props The properties object containing info about the ROM @param props The properties object containing info about the ROM
@return On success, the PNG file has been saved to 'filename',
otherwise a const char* exception is thrown containing a
more detailed error message.
*/ */
string saveImage(const string& filename, const FrameBuffer& framebuffer, void saveImage(const string& filename, const Properties& props);
const Properties& props);
/** /**
Save the current TIA image to a PNG file using data directly from Save the current TIA image to a PNG file using data directly from
the TIA framebuffer. No filtering or scaling will be included. the internal TIA buffer. No filtering or scaling will be included.
@param filename The filename to save the PNG image @param filename The filename to save the PNG image
@param framebuffer The framebuffer containing the image data @param tia Source of the raw TIA data
@param mediasrc Source of the raw TIA data
@param props The properties object containing info about the ROM @param props The properties object containing info about the ROM
@return On success, the PNG file has been saved to 'filename',
otherwise a const char* exception is thrown containing a
more detailed error message.
*/ */
string saveImage(const string& filename, const FrameBuffer& framebuffer, void saveImage(const string& filename, const TIA& tia, const Properties& props);
const TIA& tia, const Properties& props);
private: private:
const FrameBuffer& myFB;
// The following data remains between invocations of allocateStorage, // The following data remains between invocations of allocateStorage,
// and is only changed when absolutely necessary. // and is only changed when absolutely necessary.
typedef struct { typedef struct {
@ -100,6 +105,9 @@ class PNGLibrary
The method fills the 'ReadInfo' struct with valid memory locations The method fills the 'ReadInfo' struct with valid memory locations
dependent on the given dimensions. If memory has been previously dependent on the given dimensions. If memory has been previously
allocated and it can accommodate the given dimensions, it is used directly. allocated and it can accommodate the given dimensions, it is used directly.
@param iwidth The width of the PNG image
@param iheight The height of the PNG image
*/ */
bool allocateStorage(png_uint_32 iwidth, png_uint_32 iheight); bool allocateStorage(png_uint_32 iwidth, png_uint_32 iheight);
@ -107,18 +115,17 @@ class PNGLibrary
Scale the PNG data from 'ReadInfo' into the FBSurface. For now, scaling Scale the PNG data from 'ReadInfo' into the FBSurface. For now, scaling
is done on integer boundaries only (ie, 1x, 2x, etc up or down). is done on integer boundaries only (ie, 1x, 2x, etc up or down).
@param fb The main framebuffer of the application
@param surface The FBSurface into which to place the PNG data @param surface The FBSurface into which to place the PNG data
*/ */
void scaleImagetoSurface(const FrameBuffer& fb, FBSurface& surface); void scaleImagetoSurface(FBSurface& surface);
string saveBufferToPNG(ofstream& out, uInt8* buffer, /**
uInt32 width, uInt32 height, Write PNG tEXt chunks to the image.
const Properties& props, */
const string& effectsInfo); void writeComments(png_structp png_ptr, png_infop info_ptr,
void writePNGChunk(ofstream& out, const char* type, uInt8* data, int size); const Properties& props);
void writePNGText(ofstream& out, const string& key, const string& text);
/** PNG library callback functions */
static void png_read_data(png_structp ctx, png_bytep area, png_size_t size); static void png_read_data(png_structp ctx, png_bytep area, png_size_t size);
static void png_write_data(png_structp ctx, png_bytep area, png_size_t size); static void png_write_data(png_structp ctx, png_bytep area, png_size_t size);
static void png_io_flush(png_structp ctx); static void png_io_flush(png_structp ctx);

View File

@ -771,8 +771,7 @@ int PromptWidget::vprintf(const char *format, va_list argptr)
void PromptWidget::putchar(int c) void PromptWidget::putchar(int c)
{ {
putcharIntern(c); putcharIntern(c);
setDirty(); draw();
setDirty(); draw(); // FIXME - not nice to redraw the full console just for one char!
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -

View File

@ -153,7 +153,7 @@ void EventHandler::reset(State state)
// We wait a little while, since 'hold' events may be present, and we want // We wait a little while, since 'hold' events may be present, and we want
// time for the ROM to process them // time for the ROM to process them
if(state == S_EMULATE) if(state == S_EMULATE)
SDL_AddTimer(500, resetEventsCallback, (void*)this); //FIXSDL SDL_AddTimer(500, resetEventsCallback, (void*)this);
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
@ -1907,26 +1907,39 @@ void EventHandler::takeSnapshot(uInt32 number)
// Now create a PNG snapshot // Now create a PNG snapshot
if(myOSystem.settings().getBool("ss1x")) if(myOSystem.settings().getBool("ss1x"))
{ {
string msg = string message = "Snapshot saved";
myOSystem.png().saveImage(filename, myOSystem.frameBuffer(), try
myOSystem.console().tia(), {
myOSystem.png().saveImage(filename, myOSystem.console().tia(),
myOSystem.console().properties()); myOSystem.console().properties());
}
catch(const char* msg)
{
message = msg;
}
if(showmessage) if(showmessage)
myOSystem.frameBuffer().showMessage(msg); myOSystem.frameBuffer().showMessage(message);
} }
else else
{ {
// Make sure we have a 'clean' image, with no onscreen messages // Make sure we have a 'clean' image, with no onscreen messages
myOSystem.frameBuffer().enableMessages(false); myOSystem.frameBuffer().enableMessages(false);
string msg = string message = "Snapshot saved";
myOSystem.png().saveImage(filename, myOSystem.frameBuffer(), try
myOSystem.console().properties()); {
myOSystem.png().saveImage(filename, myOSystem.console().properties());
}
catch(const char* msg)
{
message = msg;
}
// Re-enable old messages // Re-enable old messages
myOSystem.frameBuffer().enableMessages(true); myOSystem.frameBuffer().enableMessages(true);
if(showmessage) if(showmessage)
myOSystem.frameBuffer().showMessage(msg); myOSystem.frameBuffer().showMessage(message);
} }
} }

View File

@ -535,14 +535,6 @@ void FrameBuffer::refresh()
case EventHandler::S_PAUSE: case EventHandler::S_PAUSE:
invalidate(); invalidate();
drawTIA(); drawTIA();
#if 0 // FIXME: eliminate stuttering in TIA mode; do we really need this?
if(isDoubleBuffered())
{
postFrameUpdate();
invalidate();
drawTIA();
}
#endif
break; break;
case EventHandler::S_MENU: case EventHandler::S_MENU:

View File

@ -352,15 +352,16 @@ class FrameBuffer
virtual bool isDoubleBuffered() const = 0; virtual bool isDoubleBuffered() const = 0;
/** /**
This method is called to get the specified scanline data from the This method is called to get the specified ARGB data from the viewable
viewable FrameBuffer area. Note that this isn't the same as any FrameBuffer area. Note that this isn't the same as any internal
internal surfaces that may be in use; it should return the actual surfaces that may be in use; it should return the actual data as it
data as it is currently seen onscreen. is currently seen onscreen.
@param row The row we are looking for @param rect The bounding rectangle for the buffer
@param data The actual pixel data (in bytes) @param buffer The actual pixel data in ARGB8888 format
@param pitch The pitch (in bytes) for the pixel data
*/ */
virtual void scanline(uInt32 row, uInt8* data) const = 0; virtual void readPixels(const GUI::Rect& rect, uInt8* buffer, uInt32 pitch) const = 0;
protected: protected:
/** /**

View File

@ -219,7 +219,7 @@ bool OSystem::create()
Random::setSystem(this); Random::setSystem(this);
// Create PNG handler // Create PNG handler
myPNGLib = new PNGLibrary(); myPNGLib = new PNGLibrary(*myFrameBuffer);
// Create ZIP handler // Create ZIP handler
myZipHandler = new ZipHandler(); myZipHandler = new ZipHandler();

View File

@ -99,7 +99,7 @@ void RomInfoWidget::parseProperties()
// Initialize to empty properties entry // Initialize to empty properties entry
mySurfaceErrorMsg = ""; mySurfaceErrorMsg = "";
mySurfaceIsValid = false; mySurfaceIsValid = true;
myRomInfo.clear(); myRomInfo.clear();
// Get a valid filename representing a snapshot file for this rom // Get a valid filename representing a snapshot file for this rom
@ -109,8 +109,7 @@ void RomInfoWidget::parseProperties()
// Read the PNG file // Read the PNG file
try try
{ {
mySurfaceIsValid = instance().png().loadImage(filename, *mySurface);
instance().png().loadImage(filename, instance().frameBuffer(), *mySurface);
} }
catch(const char* msg) catch(const char* msg)
{ {