Source

love / src / modules / image / magpie / ddsHandler.cpp

/**
 * Copyright (c) 2006-2013 LOVE Development Team
 *
 * This software is provided 'as-is', without any express or implied
 * warranty.  In no event will the authors be held liable for any damages
 * arising from the use of this software.
 *
 * Permission is granted to anyone to use this software for any purpose,
 * including commercial applications, and to alter it and redistribute it
 * freely, subject to the following restrictions:
 *
 * 1. The origin of this software must not be misrepresented; you must not
 *    claim that you wrote the original software. If you use this software
 *    in a product, an acknowledgment in the product documentation would be
 *    appreciated but is not required.
 * 2. Altered source versions must be plainly marked as such, and must not be
 *    misrepresented as being the original software.
 * 3. This notice may not be removed or altered from any source distribution.
 **/

#include "ddsHandler.h"

#include <algorithm>

namespace love
{
namespace image
{
namespace magpie
{

bool ddsHandler::canParse(const filesystem::FileData *data)
{
	std::string ext = data->getExtension();
	std::transform(ext.begin(), ext.end(), ext.begin(), tolower);

	if (ext.compare("dds") != 0)
		return false;

	return dds::isCompressedDDS(data->getData(), data->getSize());
}

CompressedData::TextureType ddsHandler::parse(filesystem::FileData *data, std::vector<CompressedData::SubImage> &images)
{
	if (!dds::isDDS(data->getData(), data->getSize()))
		throw love::Exception("Could not decode compressed data (not a DDS file?)");

	CompressedData::TextureType textype = CompressedData::TYPE_UNKNOWN;

	try
	{
		// Attempt to parse the dds file.
		dds::Parser parser(data->getData(), data->getSize());

		textype = convertFormat(parser.getFormat());

		if (textype == CompressedData::TYPE_UNKNOWN)
			throw love::Exception("Could not parse compressed data: Unsupported format.");

		if (parser.getMipmapCount() == 0)
			throw love::Exception("Could not parse compressed data: No readable texture data.");

		// Copy the parsed mipmap levels from the FileData to our CompressedData.
		for (size_t i = 0; i < parser.getMipmapCount(); i++)
		{
			// Fetch the data for this mipmap level.
			const dds::Image *img = parser.getImageData(i);

			CompressedData::SubImage mip;

			mip.width = img->width;
			mip.height = img->height;
			mip.size = img->dataSize;

			// Copy the mipmap image from the FileData.
			mip.data = new uint8[mip.size];
			memcpy(mip.data, img->data, mip.size);

			images.push_back(mip);
		}
	}
	catch (std::exception &e)
	{
		// Clean up any newly allocated heap memory before throwing.
		for (size_t i = 0; i < images.size(); i++)
			delete[] images[i].data;

		throw love::Exception(e.what());
	}

	return textype;
}

CompressedData::TextureType ddsHandler::convertFormat(dds::Format ddsformat)
{
	switch (ddsformat)
	{
	case dds::FORMAT_DXT1:
		return CompressedData::TYPE_DXT1;
	case dds::FORMAT_DXT3:
		return CompressedData::TYPE_DXT3;
	case dds::FORMAT_DXT5:
		return CompressedData::TYPE_DXT5;
	case dds::FORMAT_BC5:
		return CompressedData::TYPE_BC5;
	case dds::FORMAT_BC5s:
		return CompressedData::TYPE_BC5s;
	default:
		return CompressedData::TYPE_UNKNOWN;
	}

	return CompressedData::TYPE_UNKNOWN;
}

} // magpie
} // image
} // love
Tip: Filter by directory path e.g. /media app.js to search for public/media/app.js.
Tip: Use camelCasing e.g. ProjME to search for ProjectModifiedEvent.java.
Tip: Filter by extension type e.g. /repo .js to search for all .js files in the /repo directory.
Tip: Separate your search with spaces e.g. /ssh pom.xml to search for src/ssh/pom.xml.
Tip: Use ↑ and ↓ arrow keys to navigate and return to view the file.
Tip: You can also navigate files with Ctrl+j (next) and Ctrl+k (previous) and view the file with Ctrl+o.
Tip: You can also navigate files with Alt+j (next) and Alt+k (previous) and view the file with Alt+o.