2016-03-19 01:48:41 +00:00
|
|
|
#include "texture_loader_pkm.h"
|
|
|
|
#include "os/file_access.h"
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
struct ETC1Header {
|
2017-03-18 23:36:26 +00:00
|
|
|
char tag[6]; // "PKM 10"
|
|
|
|
uint16_t format; // Format == number of mips (== zero)
|
|
|
|
uint16_t texWidth; // Texture dimensions, multiple of 4 (big-endian)
|
|
|
|
uint16_t texHeight;
|
|
|
|
uint16_t origWidth; // Original dimensions (big-endian)
|
|
|
|
uint16_t origHeight;
|
2016-03-19 01:48:41 +00:00
|
|
|
};
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
RES ResourceFormatPKM::load(const String &p_path, const String &p_original_path, Error *r_error) {
|
2016-03-19 01:48:41 +00:00
|
|
|
|
|
|
|
if (r_error)
|
2017-03-18 23:36:26 +00:00
|
|
|
*r_error = ERR_CANT_OPEN;
|
2016-03-19 01:48:41 +00:00
|
|
|
|
|
|
|
Error err;
|
2017-03-18 23:36:26 +00:00
|
|
|
FileAccess *f = FileAccess::open(p_path, FileAccess::READ, &err);
|
2016-03-19 01:48:41 +00:00
|
|
|
if (!f)
|
|
|
|
return RES();
|
|
|
|
|
|
|
|
FileAccessRef fref(f);
|
|
|
|
if (r_error)
|
2017-03-18 23:36:26 +00:00
|
|
|
*r_error = ERR_FILE_CORRUPT;
|
2016-03-19 01:48:41 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
ERR_EXPLAIN("Unable to open PKM texture file: " + p_path);
|
|
|
|
ERR_FAIL_COND_V(err != OK, RES());
|
2016-03-19 01:48:41 +00:00
|
|
|
|
|
|
|
// big endian
|
|
|
|
f->set_endian_swap(true);
|
|
|
|
|
|
|
|
ETC1Header h;
|
2017-03-18 23:36:26 +00:00
|
|
|
ERR_EXPLAIN("Invalid or Unsupported PKM texture file: " + p_path);
|
|
|
|
f->get_buffer((uint8_t *)&h.tag, sizeof(h.tag));
|
|
|
|
if (strncmp(h.tag, "PKM 10", sizeof(h.tag)))
|
2016-03-19 01:48:41 +00:00
|
|
|
ERR_FAIL_V(RES());
|
|
|
|
|
|
|
|
h.format = f->get_16();
|
|
|
|
h.texWidth = f->get_16();
|
|
|
|
h.texHeight = f->get_16();
|
|
|
|
h.origWidth = f->get_16();
|
|
|
|
h.origHeight = f->get_16();
|
|
|
|
|
|
|
|
DVector<uint8_t> src_data;
|
|
|
|
|
|
|
|
uint32_t size = h.texWidth * h.texHeight / 2;
|
|
|
|
src_data.resize(size);
|
|
|
|
DVector<uint8_t>::Write wb = src_data.write();
|
2017-03-18 23:36:26 +00:00
|
|
|
f->get_buffer(wb.ptr(), size);
|
|
|
|
wb = DVector<uint8_t>::Write();
|
2016-03-19 01:48:41 +00:00
|
|
|
|
|
|
|
int mipmaps = h.format;
|
|
|
|
int width = h.origWidth;
|
|
|
|
int height = h.origHeight;
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
Image img(width, height, mipmaps, Image::FORMAT_ETC, src_data);
|
2016-03-19 01:48:41 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
Ref<ImageTexture> texture = memnew(ImageTexture);
|
2016-03-19 01:48:41 +00:00
|
|
|
texture->create_from_image(img);
|
|
|
|
|
|
|
|
if (r_error)
|
2017-03-18 23:36:26 +00:00
|
|
|
*r_error = OK;
|
2016-03-19 01:48:41 +00:00
|
|
|
|
|
|
|
return texture;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ResourceFormatPKM::get_recognized_extensions(List<String> *p_extensions) const {
|
|
|
|
|
|
|
|
p_extensions->push_back("pkm");
|
|
|
|
}
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
bool ResourceFormatPKM::handles_type(const String &p_type) const {
|
2016-03-19 01:48:41 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
return ObjectTypeDB::is_type(p_type, "Texture");
|
2016-03-19 01:48:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
String ResourceFormatPKM::get_resource_type(const String &p_path) const {
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
if (p_path.extension().to_lower() == "pkm")
|
2016-03-19 01:48:41 +00:00
|
|
|
return "ImageTexture";
|
|
|
|
return "";
|
|
|
|
}
|