add DXT5 to DXT1 conversion tool

This commit is contained in:
Andrew Cassidy 2021-01-25 21:17:59 -08:00
parent c89d94717d
commit b6aa96cca3
4 changed files with 197 additions and 61 deletions

View File

@ -1,4 +1,5 @@
import os
import subprocess
import struct
import re
import math
@ -74,7 +75,8 @@ class DDSPixelFlags(enum.IntFlag):
texture_formats = {
'DXT1' : s3tc.DXT1Texture
'DXT1' : s3tc.DXT1Texture,
'DXT5' : s3tc.DXT5Texture
}
class DDSFile:

View File

@ -1,43 +1,66 @@
#!/usr/bin/env python3
"""Checks any number of dds files for common issues"""
import sys
import os
import os.path
import enum
import argparse
import dds
parser = argparse.ArgumentParser(description="Checks any number of dds files for common issues, including formats not supported by KSP, and DXT5 textures that don't use the alpha channel.")
class Check(enum.Enum):
"""Kinds of check to perform"""
TRANPARENCY = 1
FORMAT = 2
parser = argparse.ArgumentParser(description= __doc__)
parser.add_argument('files', type=str, nargs='*', help = "input dds files")
modes = parser.add_mutually_exclusive_group()
modes.add_argument('--transparency', '-t', dest='mode', action='store_const', const="transparency", default="none", help = "Generate a list of files that fail the transparency check")
modes.add_argument('--format', '-f', dest='mode', action='store_const', const="format", default="none", help = "Generate a list of files that fail the format check")
parser.add_argument('--transparency', '-t',
action='append_const', dest='checks', const=Check.TRANPARENCY,
help = "Check textures for unnecessary transparency map")
parser.add_argument('--format', '-f',
action='append_const', dest='checks', const=Check.FORMAT,
help = "Check texture formats")
parser.add_argument('--all', '-a',
action='store_const', dest='checks', const=[Check.TRANPARENCY, Check.FORMAT],
help = "Perform all checks")
parser.add_argument('--convertcmd', type=str, metavar='CMD', default=dds.convertcmd, help="name of imagemagick's convert tool (default: %(default)s)")
parser.add_argument('--infocmd', type=str, metavar='CMD', default=dds.infocmd, help="name of the nvidia dds info tool (default: %(default)s)")
parser.add_argument('--formats', nargs='+', default = ['DXT1', 'DXT5'],
help="Valid texture formats to check against (default: %(default)s)")
parser.add_argument('--convertcmd', type=str, metavar='CMD', default=dds.convertcmd,
help="name of imagemagick's convert tool (default: %(default)s)")
args = parser.parse_args()
dds.convertcmd = args.convertcmd
dds.infocmd = args.infocmd
for argv in args.files:
file = os.path.abspath(argv)
info = dds.nvinfo(file)
format = info["format"]
alpha = dds.alpha(file)
if format == "DXT1":
pass
elif format == "DXT5":
if alpha > 254:
if args.mode == 'none':
print(f'[{argv}]: Image is DXT5 but has no alpha channel')
elif args.mode == 'transparency':
print(file)
def printerror(path, shortpath, message):
"""either prints a message or just the path depending on if stdout is a tty"""
if sys.stdout.isatty():
print(f'[{shortpath}]: {message}')
else:
if args.mode == 'none':
print(f'[{argv}]: incompatible format')
elif args.mode == 'format':
print(file)
sys.stdout.write(path+'\n')
def checkfile(shortpath, checks, formats):
"""check a single dds file for common issues"""
path = os.path.abspath(shortpath)
with open(path, 'rb') as file:
ddsfile = dds.DDSFile()
ddsfile.read_header(file)
four_cc = ddsfile.four_cc
if four_cc == '':
four_cc = "uncompressed"
alpha = dds.alpha(path)
if Check.TRANPARENCY in checks and four_cc == 'DXT5' and alpha > 254:
printerror(path, shortpath, 'Image is DXT5 but has no alpha channel')
if Check.FORMAT in checks and four_cc not in formats:
printerror(path, shortpath, f'Incompatible format: {four_cc}')
for shortpath in args.files:
checkfile(shortpath, args.checks, args.formats)

40
Scripts/dxt5to1.py Executable file
View File

@ -0,0 +1,40 @@
#!/usr/bin/env python3
"""converts DXT5 dds files to DXT1 without generation loss."""
import sys
import os
import os.path
import argparse
import dds
parser = argparse.ArgumentParser(
description=__doc__)
parser.add_argument('files', type=str, nargs='*',
help = "input dds files. Paths are read from stdin if none are given")
args = parser.parse_args()
if len(args.files) < 1 and not sys.stdin.isatty():
args.files = sys.stdin.read().splitlines()
for argv in args.files:
path = os.path.abspath(argv)
# read DDS file
with open(path, 'rb') as file:
ddsfile = dds.DDSFile(file)
# check its actually dxt5
if ddsfile.four_cc != 'DXT5':
print(f'[{argv}] file is not DXT5, aborting.')
continue
print(f'[{argv}] converting to DXT1')
# make modifications
ddsfile.four_cc = 'DXT1'
ddsfile.textures = [tex.to_dxt1() for tex in ddsfile.textures]
ddsfile.linear_size //= 2
# write DDS file
with open(path, 'wb') as file:
ddsfile.write(file)

View File

@ -1,18 +1,102 @@
import struct
import math
import operator
from functools import reduce
from color import Color
def byte_slice(byte):
return [byte & 0x03,
(byte & 0x0C) >> 2,
(byte & 0x30) >> 4,
(byte & 0xC0) >> 6]
def bit_slice(value, size, count):
mask = (2 ** size) - 1
return [(value >> offset) & mask for offset in range(0, size * count, size)]
def byte_unslice(indices):
return indices[0] | (indices[1] << 2) | (indices[2] << 4) | (indices[3] << 6)
def bit_merge(values, size):
offsets = range(0, len(values) * size, size)
return reduce(operator.__or__, map(operator.lshift, values, offsets))
class DXT1Texture:
def __init__(self, width = 4, height = 4, file = None):
def triple_slice(triplet):
values = bit_slice(bit_merge(triplet, 8), 3, 8)
return [values[0:4], values[4:8]]
def triple_merge(rows):
values = rows[0] + rows[1]
return bit_slice(bit_merge(values, 3), 8, 3)
class DXT1Block:
size = 8
def __init__(self, file = None):
self.color0 = Color()
self.color1 = Color()
self.indices = [[0] * 4] * 4
if file:
self.read(file)
def __repr__(self):
return repr(self.__dict__)
def __str__(self):
return f'color0: {str(self.color0)} color1: {str(self.color1)}, indices:{self.indices}'
def read(self, file):
block = struct.unpack_from('<2H4B', file.read(self.size))
self.color0 = Color.from_565(block[0])
self.color1 = Color.from_565(block[1])
self.indices = [bit_slice(row, 2, 4) for row in block[2:6]]
def write(self, file):
file.write(struct.pack('<2H4B',
self.color0.to_565(), self.color1.to_565(),
*(bit_merge(row, 2) for row in self.indices)))
class DXT5Block:
size = 16
def __init__(self, file = None):
self.color0 = Color()
self.color1 = Color()
self.indices = [[0] * 4] * 4
self.alpha0 = 1
self.alpha1 = 1
self.alpha_indices = [[0] * 4] * 4
if file:
self.read(file)
def __repr__(self):
return repr(self.__dict__)
def read(self, file):
block = struct.unpack_from('<2B6B2H4B', file.read(self.size))
self.alpha0 = block[0] / 0xFF
self.alpha1 = block[1] / 0xFF
self.alpha_indices = triple_slice(block[2:5]) + triple_slice(block[5:8])
self.color0 = Color.from_565(block[8])
self.color1 = Color.from_565(block[9])
self.indices = [bit_slice(row, 2, 4) for row in block[10:14]]
def write(self, file):
file.write(struct.pack('<2B6B2H4B',
int(self.alpha0 * 0xFF), int(self.alpha1 * 0xFF),
*triple_merge(self.alpha_indices[0:2]),
*triple_merge(self.alpha_indices[2:4]),
self.color0.to_565(), self.color1.to_565(),
*(bit_merge(row, 2) for row in self.indices)))
def to_dxt1(self):
dxt1 = DXT1Block()
dxt1.color0 = self.color0
dxt1.color1 = self.color1
dxt1.indices = self.indices
return dxt1
class BlockTexture:
def __init__(self, block_type, width = 4, height = 4, file = None):
self.block_type = block_type
self.width = width
self.height = height
self.block_width = math.ceil(width / 4)
@ -28,35 +112,22 @@ class DXT1Texture:
def read(self, file):
for x in range(self.block_count):
self.blocks.append(DXT1Block(file))
self.blocks.append(self.block_type(file))
def write(self, file):
for block in self.blocks:
block.write(file)
class DXT1Block:
size = 8
class DXT1Texture(BlockTexture):
def __init__(self, width = 4, height = 4, file = None):
super().__init__(DXT1Block, width, height, file)
def __init__(self, file = None):
self.color0 = Color()
self.color1 = Color()
self.indices = [[0] * 4] * 4
class DXT5Texture(BlockTexture):
def __init__(self, width = 4, height = 4, file = None):
super().__init__(DXT5Block, width, height, file)
if file:
self.read(file)
def __repr__(self):
return f'color0: ({repr(self.color0)}) color1: ({repr(self.color1)}), indices:{self.indices}'
def __str__(self):
return f'color0: {str(self.color0)} color1: {str(self.color1)}, indices:{self.indices}'
def read(self, file):
block = struct.unpack_from('<HHBBBB', file.read(self.size))
self.color0 = Color.from_565(block[0])
self.color1 = Color.from_565(block[1])
self.indices = list(map(byte_slice, block[2:6]))
def write(self, file):
file.write(struct.pack('<HHBBBB', self.color0.to_565(), self.color1.to_565(), *map(byte_unslice, self.indices)))
def to_dxt1(self):
dxt1 = DXT1Texture(self.width, self.height)
dxt1.blocks = [block.to_dxt1() for block in self.blocks]
# print(f'size:{dxt1.width}x{dxt1.height}, {len(dxt1.blocks)} blocks')
return dxt1