// Copyright (c) 2005, Rodrigo Braz Monteiro // All rights reserved. // // Redistribution and use in source and binary forms, with or without // modification, are permitted provided that the following conditions are met: // // * Redistributions of source code must retain the above copyright notice, // this list of conditions and the following disclaimer. // * Redistributions in binary form must reproduce the above copyright notice, // this list of conditions and the following disclaimer in the documentation // and/or other materials provided with the distribution. // * Neither the name of the Aegisub Group nor the names of its contributors // may be used to endorse or promote products derived from this software // without specific prior written permission. // // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" // AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE // ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE // LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR // CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF // SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS // INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN // CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) // ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE // POSSIBILITY OF SUCH DAMAGE. // // ----------------------------------------------------------------------------- // // AEGISUB // // Website: http://aegisub.cellosoft.com // Contact: mailto:zeratul@cellosoft.com // /////////// // Headers #include #include #include "prs_file.h" #include "prs_entry.h" #include "prs_image.h" #include "prs_display.h" /////////////// // Constructor PRSFile::PRSFile () { } ////////////// // Destructor PRSFile::~PRSFile() { Reset(); } ////////////// // Reset file void PRSFile::Reset() { // Clear list of entries std::list::iterator cur; for (cur=entryList.begin();cur!=entryList.end();cur++) { delete *cur; } entryList.clear(); } //////// // Save void PRSFile::Save(std::string path) { // I'm using C's stdio instead of C++'s fstream because I feel that fstream is // pretty lame for binary data, so I need to make sure that no exceptions are thrown from here. // TODO: Make this endianness-independent // Open file FILE *fp = fopen(path.c_str(),"wb"); if (!fp) throw "Failed to open file"; try { // Write the "PRS" (zero-terminated) string ID (4 bytes) fwrite("PRS",1,4,fp); // Write version number (4 bytes) unsigned __int32 temp = 1; fwrite(&temp,4,1,fp); // Write stream name (for future scalability, there is only one for now) // This is writen as 4 bytes for length, and then length bytes for actual name. // Since we set length to zero, the actual name string is never writen. temp = 0; fwrite(&temp,4,1,fp); // Write data blocks std::vector vec; std::list::iterator cur; for (cur=entryList.begin();cur!=entryList.end();cur++) { // Data blocks take care of writing themselves // All of them start with a 4-byte string identifier, and a 4-byte length identifier // A decoder can (and should!) ignore any block that it doesn't recognize vec.resize(0); (*cur)->WriteData(vec); fwrite(&vec[0],1,vec.size(),fp); } } // Rethrow exceptions catch (...) { fclose(fp); throw; } // Close file fclose(fp); } //////// // Load void PRSFile::Load(std::string path, bool reset) { // Reset first, if requested if (reset) Reset(); // Open file FILE *fp = fopen(path.c_str(),"rb"); if (!fp) throw "Failed to open file"; try { // Read first four bytes char buf[5]; buf[4] = 0; fread(buf,1,4,fp); if (strcmp(buf,"PRS") != 0) throw "Invalid file type."; // Read version number unsigned __int32 temp = 0; fread(&temp,4,1,fp); if (temp != 1) throw "Invalid version."; // Read stream name length fread(&temp,4,1,fp); // Read stream name if (temp > 0) { char *streamName = new char[temp+1]; fread(streamName,1,temp,fp); // We don't need it, so delete afterwards delete streamName; } // Temporary vector std::vector vec; // Read data blocks while (!feof(fp)) { // Read identifier and size fread(buf,1,4,fp); fread(&temp,4,1,fp); // Image block if (strcmp(buf,"IMG") == 0) { // Read data vec.resize(temp); fread(&vec[0],1,temp,fp); // Create object PRSImage *img = new PRSImage; img->ReadData(vec); AddEntry(img); } // Display block else if (strcmp(buf,"DSP") == 0) { // Read data vec.resize(temp); fread(&vec[0],1,temp,fp); // Create object PRSDisplay *disp = new PRSDisplay; disp->ReadData(vec); AddEntry(disp); } // Unknown block, ignore it else { fseek(fp,temp,SEEK_CUR); } } } // Rethrow exceptions catch (...) { fclose(fp); throw; } // Close file fclose(fp); } ///////////// // Add entry void PRSFile::AddEntry(PRSEntry *entry) { entryList.push_back(entry); }