Blob Blame Raw


// TnzCore includes
#include "tsystem.h"
#include "trasterimage.h"
#include "tiio.h"
#include "timageinfo.h"
#include "tcontenthistory.h"

// TnzLib includes
#include "toonz/txshleveltypes.h"
#include "toonz/imagemanager.h"
#include "toonz/toonzscene.h"
#include "toonz/levelproperties.h"
#include "toonz/preferences.h"
#include "toonz/sceneproperties.h"

#include "toonz/levelupdater.h"

//*****************************************************************************************
//    Local namespace stuff
//*****************************************************************************************

namespace
{

inline bool supportsRandomAccess(const TFilePath &fp)
{
	const std::string &type = fp.getType();
	return type == "tlv" || // TLVs do support random access
		   //type == "pli" ||                                       // PLIs... I thought they would - but no :(
		   //type == "mov" ||                                       // MOVs are 'on the way' to support it... for now, no
		   fp.getDots() == ".."; // Multi-file levels of course do
}

//-----------------------------------------------------------------------------

void enforceBpp(TPropertyGroup *pg, int bpp, bool upgradeOnly)
{
	// Most properties have a "Bits Per Pixel" property. Enforce the M there in case.
	TEnumProperty *bppProp = (TEnumProperty *)pg->getProperty("Bits Per Pixel");
	if (bppProp) {
		typedef TEnumProperty::Range Range;
		const Range &range = bppProp->getRange();

		// Retrieve current index
		int idx = bppProp->getIndex();

		// Search for a suitable 32-bit or 64-bit value
		int currentBpp = upgradeOnly ? std::stoi(bppProp->getValueAsString()) : 0;
		int targetBpp = (std::numeric_limits<int>::max)(), targetIdx = -1;

		int i, count = (int)range.size();
		for (i = 0; i < count; ++i) {
			int bppEntry = std::stoi(range[i]);
			if ((bppEntry % bpp == 0) && currentBpp <= bppEntry && bppEntry < targetBpp)
				targetBpp = bppEntry, targetIdx = i;
		}

		if (targetIdx >= 0)
			bppProp->setIndex(targetIdx);
	}

	// Some properties have an "Alpha Channel" TBoolProperty (PNGs, currently). In case, check that.
	if (bpp % 32 == 0) {
		TBoolProperty *alphaProp = (TBoolProperty *)pg->getProperty("Alpha Channel");
		if (alphaProp)
			alphaProp->setValue(true);
	}
}

} // namespace

//*****************************************************************************************
//    LevelUpdater implementation
//*****************************************************************************************

LevelUpdater::LevelUpdater()
	: m_pg(0), m_inputLevel(0), m_currIdx(0), m_imageInfo(0), m_usingTemporaryFile(false), m_opened(false)
{
}

//-----------------------------------------------------------------------------

LevelUpdater::LevelUpdater(TXshSimpleLevel *sl)
	: m_pg(0), m_inputLevel(0), m_imageInfo(0), m_currIdx(0), m_opened(false), m_usingTemporaryFile(false)
{
	open(sl);
}

//-----------------------------------------------------------------------------

LevelUpdater::LevelUpdater(const TFilePath &fp, TPropertyGroup *lwProperties)
	: m_pg(0), m_inputLevel(0), m_imageInfo(0), m_currIdx(0), m_opened(false), m_usingTemporaryFile(false)
{
	open(fp, lwProperties);
}

//-----------------------------------------------------------------------------

LevelUpdater::~LevelUpdater()
{
	// Please, observe that the try-catch below here is NOT OPTIONAL.
	// IT IS AN ERROR TO THROW INSIDE A DESTRUCTOR. EVER.
	// Doing so damages the stack unwinding process - namely, it interferes
	// with the destruction of OTHER objects going out of scope.
	// C++ does NOT react well to that (ie could terminate() the process).

	try {
		close();
	} catch (...) {
	}
}

//-----------------------------------------------------------------------------

void LevelUpdater::reset()
{
	m_lw = TLevelWriterP();
	m_lwPath = TFilePath();

	m_lr = TLevelReaderP();
	m_inputLevel = TLevelP();
	m_sl = TXshSimpleLevelP();

	delete m_pg;
	m_pg = 0;

	if (m_imageInfo) {
		delete m_imageInfo->m_properties;
		delete m_imageInfo;
		m_imageInfo = 0;
	}

	m_fids.clear();
	m_currIdx = 0;

	m_usingTemporaryFile = false;
	m_opened = false;
}

//-----------------------------------------------------------------------------

void LevelUpdater::buildSourceInfo(const TFilePath &fp)
{
	try {
		m_lr = TLevelReaderP(fp);
		assert(m_lr);

		m_lr->enableRandomAccessRead(true); // Movie files are intended with a constant fps
											// should be made the default... TODO!
		m_inputLevel = m_lr->loadInfo();

		const TImageInfo *info = m_lr->getImageInfo();
		if (info) {
			m_imageInfo = new TImageInfo(*info); // Clone the info. The originals are owned by the reader.
			if (info->m_properties)
				m_imageInfo->m_properties = info->m_properties->clone(); // Same for these (unfortunately, TImageInfo is currently
																		 // no more than a struct...)
		}
	} catch (...) {
		// The level exists but could not be read.
		// Allowing write to a surviving temporary in this case...

		m_lr = TLevelReaderP();
		m_inputLevel = TLevelP(0);

		if (m_imageInfo) {
			delete m_imageInfo->m_properties;
			delete m_imageInfo;
			m_imageInfo = 0;
		}
	}
}

//-----------------------------------------------------------------------------

void LevelUpdater::buildProperties(const TFilePath &fp)
{
	// Ensure that at least the default properties for specified fp.getType() exist.
	m_pg = (m_imageInfo && m_imageInfo->m_properties) ? m_imageInfo->m_properties->clone() : Tiio::makeWriterProperties(fp.getType());

	if (!m_pg) {
		// If no suitable pg could be found, the extension must be wrong. Reset and throw.
		reset();
		throw TException("Unrecognized file format");
	}

	assert(m_pg);
}

//-----------------------------------------------------------------------------

void LevelUpdater::open(const TFilePath &fp, TPropertyGroup *pg)
{
	assert(!m_lw);

	// Find out if a corresponding level already exists on disk - in that case, load it
	bool existsLevel = TSystem::doesExistFileOrLevel(fp);
	if (existsLevel)
		buildSourceInfo(fp); // Could be !m_lr if level could not be read

	// Build Output Properties if needed
	if (pg)
		m_pg = pg->clone();
	else
		buildProperties(fp); // Throws only if not even the default properties
							 // could be found - ie, bad file type
	try {
		// Decide whether the update procedure requires a temporary file for appending
		m_usingTemporaryFile = existsLevel && !supportsRandomAccess(fp);
		if (m_usingTemporaryFile) {
			// The level requires a temporary to write frames to. Upon closing, the original level
			// is deleted and the temporary takes its place. Note that m_lw takes ownership of the properties group.
			m_lwPath = getNewTemporaryFilePath(fp);
			m_lw = TLevelWriterP(m_lwPath, m_pg->clone());

			if (m_inputLevel)
				for (TLevel::Iterator it = m_inputLevel->begin(); it != m_inputLevel->end(); ++it)
					m_fids.push_back(it->first);
		} else {
			m_lr = TLevelReaderP();					 // Release the reader. This is necessary since the
			m_lw = TLevelWriterP(fp, m_pg->clone()); // original file itself will be MODIFIED.
			m_lwPath = fp;
		}
	} catch (...) {
		// In this case, TLevelWriterP(..) failed, that object was never contructed,
		// the assignment m_lw never took place. And m_lw == 0.

		// Reset state and rethrow
		reset();
		throw;
	}

	// In case the writer saves icons inside the output level (TLV case), set the associated icon size now
	TDimension iconSize = Preferences::instance()->getIconSize();
	assert(iconSize.lx > 0 && iconSize.ly > 0);
	m_lw->setIconSize(iconSize);

	m_opened = true;
}

//-----------------------------------------------------------------------------

void LevelUpdater::open(TXshSimpleLevel *sl)
{
	assert(!m_lw);

	assert(sl && sl->getScene());
	m_sl = sl;

	const TFilePath &fp = sl->getScene()->decodeFilePath(sl->getPath());

	// Find out if a corresponding level already exists on disk - in that case, load it
	bool existsLevel = TSystem::doesExistFileOrLevel(fp);
	if (existsLevel)
		buildSourceInfo(fp); // Could be !m_lr if level could not be read

	// Build Output Properties
	buildProperties(fp); // May throw if not even the default properties could be
						 // retrieved

	// If there was no level on disk, or the level properties require the alpha channel, enforce the
	// bpp accordingly on m_pg.
	LevelProperties *levelProperties = sl->getProperties();
	assert(levelProperties);

	if (levelProperties->hasAlpha() || !existsLevel) {
		int bpp = levelProperties->hasAlpha() ? std::min(32, levelProperties->getBpp()) : levelProperties->getBpp();
		enforceBpp(m_pg, bpp, existsLevel);
	}

	// Should sl->getPalette() be enforced on m_lw too? It was not present in the old code...

	try {
		// Decide whether the update procedure requires a temporary file for appending
		m_usingTemporaryFile = existsLevel && !supportsRandomAccess(fp);
		if (m_usingTemporaryFile) {
			// The level requires a temporary to write frames to. Upon closing, the original level
			// is deleted and the temporary takes its place.
			m_lwPath = getNewTemporaryFilePath(fp);
			m_lw = TLevelWriterP(m_lwPath, m_pg->clone());
		} else {
			m_lr = TLevelReaderP();					 // Release the reader
			m_lw = TLevelWriterP(fp, m_pg->clone()); // Open for write the usual way
			m_lwPath = fp;
		}
	} catch (...) {
		// Reset state and rethrow
		reset();
		throw;
	}

	// Load the frames directly from sl
	sl->getFids(m_fids);

	// In case the writer saves icons inside the output level (TLV case), set the associated icon size now
	TDimension iconSize = Preferences::instance()->getIconSize();
	assert(iconSize.lx > 0 && iconSize.ly > 0);
	m_lw->setIconSize(iconSize);

	if (sl->getContentHistory())
		m_lw->setContentHistory(m_sl->getContentHistory()->clone());

	m_opened = true;
}

//-----------------------------------------------------------------------------

TFilePath LevelUpdater::getNewTemporaryFilePath(const TFilePath &fp)
{
	TFilePath fp2;
	int count = 1;

	for (;;) {
		fp2 = fp.withName(fp.getWideName() + L"__" + std::to_wstring(count++));
		if (!TSystem::doesExistFileOrLevel(fp2))
			break;
	}

	return fp2;
}

//-----------------------------------------------------------------------------

void LevelUpdater::addFramesTo(int endIdx)
{
	if (m_sl) {
		// The simple level case can be optimized since some level's images could already be present
		// in memory. Images are accessed through the level itself.

		for (; m_currIdx < endIdx; ++m_currIdx) {
			TImageP img = m_sl->getFullsampledFrame(m_fids[m_currIdx], ImageManager::dontPutInCache);
			assert(img);

			if (!img && m_lr) {
				// This should actually never happen. ImageManager should already ensure that img exists.
				// However, as last resort let's just look at the file too...
				img = m_lr->getFrameReader(m_fids[m_currIdx])->load();
				if (img)
					img->setPalette(m_sl->getPalette());
			}

			if (img)
				m_lw->getFrameWriter(m_fids[m_currIdx])->save(img);
		}
	} else if (m_lr) {
		// Otherwise, just look in the file directly
		for (; m_currIdx < endIdx; ++m_currIdx) {
			TImageP img = m_lr->getFrameReader(m_fids[m_currIdx])->load();

			if (img)
				m_lw->getFrameWriter(m_fids[m_currIdx])->save(img);
		}
	}
}

//-----------------------------------------------------------------------------

void LevelUpdater::update(const TFrameId &fid, const TImageP &img)
{
	// Resume open for write
	resume();

	if (!m_usingTemporaryFile) {
		// Plain random access write if supported
		m_lw->getFrameWriter(fid)->save(img);
		return;
	}

	// Otherwise, we must add every frame preceding fid, and *then* add img.
	// NOTE: This requires that the image sequence is already sorted by fid.
	addFramesTo(std::lower_bound(m_fids.begin() + m_currIdx, m_fids.end(), fid) - m_fids.begin());

	// Save the passed image. In case it overwrites a frame, erase that from the list too.
	m_lw->getFrameWriter(fid)->save(img);
	if (m_currIdx < int(m_fids.size()) && m_fids[m_currIdx] == fid)
		++m_currIdx;
}

//-----------------------------------------------------------------------------

void LevelUpdater::close()
{
	if (!m_opened)
		return;

	// Resume open for write
	resume();

	try {
		if (m_usingTemporaryFile) {
			// Add all remaining frames still in m_fids
			addFramesTo((int)m_fids.size());

			// Currently written level is temporary. It must be renamed to its originally intended path,
			// if it's possible to write there. Now, if it's writable, in particular it should be readable,
			// so m_lr should exist.

			// If not... well, the file was corrupt or something. Instead than attempting to delete it,
			// we're begin conservative - this means that no data is lost, but unfortunately temporaries
			// might pile up...
			if (m_lr) {
				TFilePath finalPath(m_lr->getFilePath()), tempPath(m_lw->getFilePath());

				// Release m_lr and m_lw - to be sure that no file is kept open while renaming.
				// NOTE: releasing m_lr and m_lw should not throw anything. As stated before, throwing
				//       in destructors is bad. I'm not sure this is actually guaranteed in Toonz, however :(
				m_lr = TLevelReaderP(), m_lw = TLevelWriterP();

				// Rename the level
				TSystem::removeFileOrLevel_throw(finalPath);
				TSystem::renameFileOrLevel_throw(finalPath, tempPath); // finalPath <- tempPath

				// If present, add known trailing files
				if (finalPath.getType() == "tlv") {
					// Palette file
					TFilePath finalPalette = finalPath.withType("tpl");
					TFilePath tempPalette = tempPath.withType("tpl");

					if (TFileStatus(finalPalette).doesExist()) {
						if (TFileStatus(tempPalette).doesExist())
							TSystem::deleteFile(finalPalette);
						TSystem::renameFile(finalPalette, tempPalette);
					}

					// History file
					TFilePath finalHistory = finalPath.withType("hst");
					TFilePath tempHistory = tempPath.withType("hst");

					if (TFileStatus(tempHistory).doesExist()) {
						if (TFileStatus(finalHistory).doesExist())
							TSystem::deleteFile(finalHistory);
						TSystem::renameFile(finalHistory, tempHistory);
					}
				}
			}

			// NOTE: If for some reason m_lr was not present and we were using a temporary file, no
			// renaming takes place. Users could see the __x temporaries and, eventually, rename them manually
			// or see what's wrong with the unwritable file.
		}

		// Reset the updater's status
		reset();
	} catch (...) {
		// Some temporary object could not be renamed. Or some remaining frame could not be added.
		// Hopefully, it was not about closing m_lr or m_lw.

		// However, we still intend to reset the updater's status before rethrowing.
		reset();
		throw;
	}
}

//-----------------------------------------------------------------------------

void LevelUpdater::flush()
{
	assert(m_opened);
	if (!m_lw)
		return;

	// In case the level writer could not be destroyed (bad, should really not throw btw),
	// reset and rethrow
	try {
		m_lw = TLevelWriterP();
	} catch (...) {
		reset();
		throw;
	}
}

//-----------------------------------------------------------------------------

void LevelUpdater::resume()
{
	assert(m_opened);
	if (m_lw)
		return;

	try {
		m_lw = TLevelWriterP(m_lwPath, m_pg->clone());
	} catch (...) {
		reset();
		throw;
	}
}