2021-01-23 03:59:00 +00:00
|
|
|
import os
|
2021-01-26 05:17:59 +00:00
|
|
|
import subprocess
|
2021-01-23 03:59:00 +00:00
|
|
|
import struct
|
2021-01-20 07:57:24 +00:00
|
|
|
import re
|
2021-01-25 08:57:47 +00:00
|
|
|
import math
|
2021-01-23 03:59:00 +00:00
|
|
|
import enum
|
2021-01-25 08:57:47 +00:00
|
|
|
import s3tc
|
2021-01-23 03:59:00 +00:00
|
|
|
|
|
|
|
if os.name == 'nt':
|
|
|
|
convertcmd = "convert.exe"
|
|
|
|
compresscmd = "nvcompress.exe"
|
|
|
|
decompresscmd = "nvdecompress.exe"
|
|
|
|
infocmd = "nvddsinfo.exe"
|
|
|
|
else:
|
|
|
|
convertcmd = "convert"
|
|
|
|
compresscmd = "nvcompress"
|
|
|
|
decompresscmd = "nvdecompress"
|
|
|
|
infocmd = "nvddsinfo"
|
2021-01-20 07:57:24 +00:00
|
|
|
|
|
|
|
def alpha(file):
|
2021-01-23 03:59:00 +00:00
|
|
|
result = subprocess.run([convertcmd, file, "-resize", "1x1",
|
|
|
|
"-format", "%[fx:int(255*a+.5)]", "info:-"], capture_output=True)
|
2021-01-20 07:57:24 +00:00
|
|
|
result.check_returncode()
|
|
|
|
return int(result.stdout)
|
|
|
|
|
2021-01-23 03:59:00 +00:00
|
|
|
|
2021-01-20 07:57:24 +00:00
|
|
|
def flip(file, output):
|
2021-01-23 03:59:00 +00:00
|
|
|
result = subprocess.run(
|
|
|
|
[convertcmd, '-flip', file, output], capture_output=True)
|
2021-01-20 07:57:24 +00:00
|
|
|
result.check_returncode()
|
2021-01-23 03:59:00 +00:00
|
|
|
|
|
|
|
|
|
|
|
def nvcompress(format, file, output, mips=True):
|
2021-01-20 10:28:17 +00:00
|
|
|
args = [format]
|
|
|
|
if not mips:
|
|
|
|
args.append('-nomips')
|
2021-01-23 03:59:00 +00:00
|
|
|
|
|
|
|
result = subprocess.run([compresscmd] + args +
|
|
|
|
[file, output], capture_output=True)
|
2021-01-20 07:57:24 +00:00
|
|
|
result.check_returncode()
|
2021-01-23 03:59:00 +00:00
|
|
|
|
|
|
|
|
2021-01-20 07:57:24 +00:00
|
|
|
def nvdecompress(file, output):
|
2021-01-20 09:40:09 +00:00
|
|
|
result = subprocess.run([decompresscmd, file, output], capture_output=True)
|
2021-01-20 07:57:24 +00:00
|
|
|
result.check_returncode()
|
|
|
|
|
2021-01-23 03:59:00 +00:00
|
|
|
|
2021-01-20 07:57:24 +00:00
|
|
|
def nvinfo(file):
|
2021-01-20 09:40:09 +00:00
|
|
|
result = subprocess.run([infocmd, file], capture_output=True)
|
2021-01-20 07:57:24 +00:00
|
|
|
result.check_returncode()
|
2021-01-23 03:59:00 +00:00
|
|
|
|
2021-01-20 07:57:24 +00:00
|
|
|
info = {
|
|
|
|
"format": re.search(r"FourCC: '(.{4})'", str(result.stdout)).group(1)
|
|
|
|
}
|
2021-01-23 03:59:00 +00:00
|
|
|
|
|
|
|
return info
|
|
|
|
|
|
|
|
class DDSFlags(enum.IntFlag):
|
|
|
|
CAPS = 0x1
|
|
|
|
HEIGHT = 0x2
|
|
|
|
WIDTH = 0x4
|
|
|
|
PITCH = 0x8
|
|
|
|
PIXEL_FORMAT = 0x1000
|
|
|
|
MIPMAPCOUNT = 0x20000
|
|
|
|
LINEAR_SIZE = 0x80000
|
|
|
|
DEPTH = 0x800000
|
|
|
|
|
|
|
|
class DDSPixelFlags(enum.IntFlag):
|
|
|
|
ALPHAPIXELS = 0x1
|
|
|
|
ALPHA = 0x2
|
|
|
|
FOURCC = 0x4
|
|
|
|
RGB = 0x40
|
|
|
|
YUV = 0x200
|
|
|
|
LUMINANCE = 0x20000
|
|
|
|
|
2021-01-25 08:57:47 +00:00
|
|
|
|
|
|
|
texture_formats = {
|
2021-01-26 05:17:59 +00:00
|
|
|
'DXT1' : s3tc.DXT1Texture,
|
|
|
|
'DXT5' : s3tc.DXT5Texture
|
2021-01-25 08:57:47 +00:00
|
|
|
}
|
|
|
|
|
2021-01-24 03:52:11 +00:00
|
|
|
class DDSFile:
|
|
|
|
def __init__(self, file = None):
|
|
|
|
if (file):
|
2021-01-25 08:57:47 +00:00
|
|
|
self.read(file)
|
|
|
|
|
|
|
|
def read(self, file):
|
|
|
|
self.read_header(file)
|
|
|
|
self.read_data(file)
|
|
|
|
|
|
|
|
def write(self, file):
|
|
|
|
self.write_header(file)
|
|
|
|
self.write_data(file)
|
2021-01-23 03:59:00 +00:00
|
|
|
|
2021-01-24 03:52:11 +00:00
|
|
|
def read_header(self, file):
|
|
|
|
# read magic bytes
|
|
|
|
if file.read(4).decode() != 'DDS ':
|
|
|
|
raise ValueError("File is not a valid DDS file: Incorrect magic bytes")
|
2021-01-23 03:59:00 +00:00
|
|
|
|
2021-01-24 03:52:11 +00:00
|
|
|
# read header
|
2021-01-25 08:57:47 +00:00
|
|
|
header = struct.unpack('<7I44x', file.read(72))
|
2021-01-23 03:59:00 +00:00
|
|
|
|
2021-01-24 03:52:11 +00:00
|
|
|
# check constant
|
|
|
|
if header[0] != 124:
|
|
|
|
raise ValueError("File is not a valid DDS file: Incorrect header size")
|
2021-01-23 03:59:00 +00:00
|
|
|
|
|
|
|
# read basic info
|
2021-01-24 03:52:11 +00:00
|
|
|
self.flags = DDSFlags(header[1])
|
|
|
|
self.height, self.width, self.linear_size, self.depth, self.mipmapcount = header[2:7]
|
2021-01-23 03:59:00 +00:00
|
|
|
|
2021-01-24 03:52:11 +00:00
|
|
|
# read pixelformat
|
2021-01-25 08:57:47 +00:00
|
|
|
pixelformat = struct.unpack('<2I4s5I', file.read(32))
|
2021-01-23 03:59:00 +00:00
|
|
|
|
2021-01-24 03:52:11 +00:00
|
|
|
# check constant
|
|
|
|
if pixelformat[0] != 32:
|
|
|
|
raise ValueError("File is not a valid DDS file: Incorrect pixelformat size")
|
2021-01-23 03:59:00 +00:00
|
|
|
|
2021-01-24 03:52:11 +00:00
|
|
|
self.pixel_flags = DDSPixelFlags(pixelformat[1])
|
|
|
|
self.four_cc = pixelformat[2].decode()
|
|
|
|
self.rgb_bit_count, self.r_bitmask, self.g_bitmask, self.b_bitmask, self.a_bitmask = pixelformat[3:8]
|
2021-01-23 03:59:00 +00:00
|
|
|
|
|
|
|
# read caps
|
2021-01-25 08:57:47 +00:00
|
|
|
self.dw_caps, self.dw_caps2, self.dw_caps3, self.dw_caps4 = struct.unpack('<4I4x', file.read(20))
|
2021-01-24 03:52:11 +00:00
|
|
|
|
|
|
|
# read DX10 header
|
|
|
|
self.is_DX10 = (self.four_cc == 'DX10')
|
|
|
|
if self.is_DX10:
|
|
|
|
self.read_dx10header(file)
|
|
|
|
|
|
|
|
def read_dx10header(self, file):
|
|
|
|
dx10header = file.read(20)
|
|
|
|
#TODO: actually do something with the DX10 data
|
|
|
|
|
|
|
|
def write_header(self, file):
|
2021-01-25 08:57:47 +00:00
|
|
|
file.write(bytes('DDS ', 'utf-8'))
|
2021-01-24 03:52:11 +00:00
|
|
|
|
|
|
|
# write header
|
|
|
|
file.write(struct.pack(
|
2021-01-25 08:57:47 +00:00
|
|
|
'<7I44x', 124, int(self.flags), self.height, self.width,
|
2021-01-24 03:52:11 +00:00
|
|
|
self.linear_size, self.depth, self.mipmapcount))
|
|
|
|
|
|
|
|
# write pixelformat
|
|
|
|
file.write(struct.pack(
|
2021-01-25 08:57:47 +00:00
|
|
|
'<2I4s5I', 32, int(self.pixel_flags), bytes(self.four_cc, 'utf-8'), self.rgb_bit_count,
|
2021-01-24 03:52:11 +00:00
|
|
|
self.r_bitmask, self.g_bitmask, self.b_bitmask, self.a_bitmask))
|
|
|
|
|
|
|
|
# write caps
|
|
|
|
file.write(struct.pack(
|
2021-01-25 08:57:47 +00:00
|
|
|
'<4I4x', self.dw_caps, self.dw_caps2, self.dw_caps3, self.dw_caps4))
|
2021-01-23 03:59:00 +00:00
|
|
|
|
2021-01-25 08:57:47 +00:00
|
|
|
#TODO: handle DX10 data
|
|
|
|
if self.is_DX10:
|
|
|
|
file.write(struct.pack('20x'))
|
|
|
|
|
|
|
|
def read_data(self, file):
|
|
|
|
self.textures = []
|
|
|
|
|
|
|
|
if DDSFlags.MIPMAPCOUNT in self.flags:
|
|
|
|
mips = self.mipmapcount
|
|
|
|
else:
|
|
|
|
mips = 1
|
|
|
|
|
|
|
|
if DDSPixelFlags.FOURCC in self.pixel_flags:
|
|
|
|
textureType = texture_formats[self.four_cc]
|
|
|
|
for mip in range(mips):
|
|
|
|
width = math.ceil(self.width / (2 ** mip))
|
|
|
|
height = math.ceil(self.height / (2 ** mip))
|
|
|
|
self.textures.append(textureType(width, height, file))
|
|
|
|
|
|
|
|
def write_data(self, file):
|
|
|
|
for texture in self.textures:
|
|
|
|
texture.write(file)
|