phaser/src/textures/parsers/PVRParser.js

221 lines
5.9 KiB
JavaScript
Raw Normal View History

2021-10-05 17:30:24 +00:00
// https://www.khronos.org/registry/webgl/extensions/WEBGL_compressed_texture_pvrtc/
// https://www.khronos.org/registry/webgl/extensions/WEBGL_compressed_texture_s3tc/
// https://www.khronos.org/registry/webgl/extensions/WEBGL_compressed_texture_etc/
// https://www.khronos.org/registry/webgl/extensions/WEBGL_compressed_texture_astc/
function GetSize (width, height, x, y, dx, dy, mult)
{
if (mult === undefined) { mult = 16; }
return Math.floor((width + x) / dx) * Math.floor((height + y) / dy) * mult;
}
function PVRTC2bppSize (width, height)
{
width = Math.max(width, 16);
height = Math.max(height, 8);
return width * height / 4;
}
function PVRTC4bppSize (width, height)
{
width = Math.max(width, 8);
height = Math.max(height, 8);
return width * height / 2;
}
function DXTEtcSmallSize (width, height)
{
return GetSize(width, height, 3, 3, 4, 4, 8);
}
function DXTEtcAstcBigSize (width, height)
{
return GetSize(width, height, 3, 3, 4, 4);
}
function ATC5x4Size (width, height)
{
return GetSize(width, height, 4, 3, 5, 4);
}
function ATC5x5Size (width, height)
{
return GetSize(width, height, 4, 4, 5, 5);
}
function ATC6x5Size (width, height)
{
return GetSize(width, height, 5, 4, 6, 5);
}
function ATC6x6Size (width, height)
{
return GetSize(width, height, 5, 5, 6, 6);
}
function ATC8x5Size (width, height)
{
return GetSize(width, height, 7, 4, 8, 5);
}
function ATC8x6Size (width, height)
{
return GetSize(width, height, 7, 5, 8, 6);
}
function ATC8x8Size (width, height)
{
return GetSize(width, height, 7, 7, 8, 8);
}
function ATC10x5Size (width, height)
{
return GetSize(width, height, 9, 4, 10, 5);
}
function ATC10x6Size (width, height)
{
return GetSize(width, height, 9, 5, 10, 6);
}
function ATC10x8Size (width, height)
{
return GetSize(width, height, 9, 7, 10, 8);
}
function ATC10x10Size (width, height)
{
return GetSize(width, height, 9, 9, 10, 10);
}
function ATC12x10Size (width, height)
{
return GetSize(width, height, 11, 9, 12, 10);
}
function ATC12x12Size (width, height)
{
return GetSize(width, height, 11, 11, 12, 12);
}
// 0: COMPRESSED_RGB_PVRTC_2BPPV1_IMG
// 1: COMPRESSED_RGBA_PVRTC_2BPPV1_IMG
// 2: COMPRESSED_RGB_PVRTC_4BPPV1_IMG
// 3: COMPRESSED_RGBA_PVRTC_4BPPV1_IMG
// 6: COMPRESSED_RGB_ETC1
// 7: COMPRESSED_RGB_S3TC_DXT1_EXT
// 8: COMPRESSED_RGBA_S3TC_DXT1_EXT
// 9: COMPRESSED_RGBA_S3TC_DXT3_EXT
// 11: COMPRESSED_RGBA_S3TC_DXT5_EXT
// 22: COMPRESSED_RGB8_ETC2
// 23: COMPRESSED_RGBA8_ETC2_EAC
// 24: COMPRESSED_RGB8_PUNCHTHROUGH_ALPHA1_ETC2
// 25: COMPRESSED_R11_EAC
// 26: COMPRESSED_RG11_EAC
// 27: COMPRESSED_RGBA_ASTC_4x4_KHR
// 28: COMPRESSED_RGBA_ASTC_5x4_KHR
// 29: COMPRESSED_RGBA_ASTC_5x5_KHR
// 30: COMPRESSED_RGBA_ASTC_6x5_KHR
// 31: COMPRESSED_RGBA_ASTC_6x6_KHR
// 32: COMPRESSED_RGBA_ASTC_8x5_KHR
// 33: COMPRESSED_RGBA_ASTC_8x6_KHR
// 34: COMPRESSED_RGBA_ASTC_8x8_KHR
// 35: COMPRESSED_RGBA_ASTC_10x5_KHR
// 36: COMPRESSED_RGBA_ASTC_10x6_KHR
// 37: COMPRESSED_RGBA_ASTC_10x8_KHR
// 38: COMPRESSED_RGBA_ASTC_10x10_KHR
// 39: COMPRESSED_RGBA_ASTC_12x10_KHR
// 40: COMPRESSED_RGBA_ASTC_12x12_KHR
var FORMATS = {
0: { sizeFunc: PVRTC2bppSize, glFormat: 0x8C01 },
1: { sizeFunc: PVRTC2bppSize, glFormat: 0x8C03 },
2: { sizeFunc: PVRTC4bppSize, glFormat: 0x8C00 },
3: { sizeFunc: PVRTC4bppSize, glFormat: 0x8C02 },
6: { sizeFunc: DXTEtcSmallSize , glFormat: 0x8D64 },
7: { sizeFunc: DXTEtcSmallSize, glFormat: 0x83F0 },
8: { sizeFunc: DXTEtcAstcBigSize, glFormat: 0x83F1 },
9: { sizeFunc: DXTEtcAstcBigSize, glFormat: 0x83F2 },
11: { sizeFunc: DXTEtcAstcBigSize, glFormat: 0x83F3 },
22: { sizeFunc: DXTEtcSmallSize , glFormat: 0x9274 },
23: { sizeFunc: DXTEtcAstcBigSize, glFormat: 0x9278 },
24: { sizeFunc: DXTEtcSmallSize, glFormat: 0x9276 },
25: { sizeFunc: DXTEtcSmallSize, glFormat: 0x9270 },
26: { sizeFunc: DXTEtcAstcBigSize, glFormat: 0x9272 },
27: { sizeFunc: DXTEtcAstcBigSize, glFormat: 0x93B0 },
28: { sizeFunc: ATC5x4Size, glFormat: 0x93B1 },
29: { sizeFunc: ATC5x5Size, glFormat: 0x93B2 },
30: { sizeFunc: ATC6x5Size, glFormat: 0x93B3 },
31: { sizeFunc: ATC6x6Size, glFormat: 0x93B4 },
32: { sizeFunc: ATC8x5Size, glFormat: 0x93B5 },
33: { sizeFunc: ATC8x6Size, glFormat: 0x93B6 },
34: { sizeFunc: ATC8x8Size, glFormat: 0x93B7 },
35: { sizeFunc: ATC10x5Size, glFormat: 0x93B8 },
36: { sizeFunc: ATC10x6Size, glFormat: 0x93B9 },
37: { sizeFunc: ATC10x8Size, glFormat: 0x93BA },
38: { sizeFunc: ATC10x10Size, glFormat: 0x93BB },
39: { sizeFunc: ATC12x10Size, glFormat: 0x93BC },
40: { sizeFunc: ATC12x12Size, glFormat: 0x93BD }
};
var PVRParser = function (data)
{
var header = new Uint32Array(data, 0, 13);
// PIXEL_FORMAT_INDEX
var pvrFormat = header[2];
var internalFormat = FORMATS[pvrFormat].glFormat;
var sizeFunction = FORMATS[pvrFormat].sizeFunc;
// MIPMAPCOUNT_INDEX
var mipmapLevels = header[11];
// WIDTH_INDEX
var width = header[7];
// HEIGHT_INDEX
var height = header[6];
// HEADER_SIZE + METADATA_SIZE_INDEX
var dataOffset = 52 + header[12];
var image = new Uint8Array(data, dataOffset);
var mipmaps = new Array(mipmapLevels);
var offset = 0;
var levelWidth = width;
var levelHeight = height;
for (var i = 0; i < mipmapLevels; i++)
{
var levelSize = sizeFunction(levelWidth, levelHeight);
mipmaps[i] = {
data: new Uint8Array(image.buffer, image.byteOffset + offset, levelSize),
width: levelWidth,
height: levelHeight
};
levelWidth = Math.max(1, levelWidth >> 1);
levelHeight = Math.max(1, levelHeight >> 1);
offset += levelSize;
}
return {
mipmaps: mipmaps,
width: width,
height: height,
internalFormat: internalFormat,
compressed: true,
generateMipmap: false
};
};
module.exports = PVRParser;