freedoom/lumps/genmidi/a2i_file.py
Mike Swanson 6eef9be73a use python3 only for building
Python 2 is very near end-of-life, and Python3-compatible changes to a
few scripts introduced compatibility problems with 2.7 again.  It went
unnoticed for me since my system symlinks "python" to "python3", but
it broke the build on systems where that symlink is still python2.  At
this point in time, I feel it is worth targetting modern Python and
forgetting about 2.7.
2019-09-06 14:43:50 -07:00

224 lines
5.4 KiB
Python

#!/usr/bin/env python3
# SPDX-License-Identifier: BSD-3-Clause
#
# Module to load A2I (AdlibTracker 2) instrument files.
#
import sys
import struct
HEADER_STRING = "_A2ins_"
class BitReader:
def __init__(self, data):
self.data = data
self.index = 0
self.byte = 0
self.byte_bit = 0
def read_byte(self):
if self.index >= len(self.data):
raise IndexError(
"Reached end of decompress stream "
+ "(%i bytes)" % len(self.data)
)
result, = struct.unpack("B", self.data[self.index : self.index + 1])
self.index += 1
return result
def read_bit(self):
if self.byte_bit <= 0:
self.byte = self.read_byte()
self.byte_bit = 7
else:
self.byte_bit -= 1
if (self.byte & (1 << self.byte_bit)) != 0:
result = 1
else:
result = 0
return result
def read_bits(self, n):
result = 0
for i in range(n):
result = (result << 1) + self.read_bit()
return result
def read_gamma(reader):
result = 1
while True:
result = (result << 1) | reader.read_bit()
if reader.read_bit() == 0:
break
return result
def decompress(data, data_len):
reader = BitReader(data)
result = []
lwm = 0
last_offset = 0
# First byte is an implied straight copy.
result.append(reader.read_byte())
while True:
if reader.read_bit():
if reader.read_bit():
if reader.read_bit():
# 111 = Copy byte from history,
# up to 15 bytes back.
# print "111 copy"
offset = reader.read_bits(4)
if offset == 0:
result.append(0)
else:
b = result[len(result) - offset]
result.append(b)
lwm = 0
else:
# print "110 copy"
# 110 = Copy 2-3 bytes from
# further back in history
offset = reader.read_byte()
count = 2 + (offset & 0x01)
offset = offset >> 1
if offset == 0:
break
index = len(result) - offset
for i in range(count):
result += result[index : index + 1]
index += 1
last_offset = offset
lwm = 1
else:
# 10 = Copy from further away...
offset = read_gamma(reader)
if lwm == 0 and offset == 2:
# print "10 copy type 1"
count = read_gamma(reader)
index = len(result) - last_offset
for i in range(count):
result += result[index : index + 1]
index += 1
else:
# print "10 copy type 2"
if lwm == 0:
offset -= 3
else:
offset -= 2
offset = offset * 256 + reader.read_byte()
count = read_gamma(reader)
if offset >= 32000:
count += 1
if offset >= 1280:
count += 1
if offset < 128:
count += 2
index = len(result) - offset
for i in range(count):
result += result[index : index + 1]
index += 1
last_offset = offset
lwm = 1
else:
# print "Single byte output"
# 0 = Straight-through byte copy.
result.append(reader.read_byte())
lwm = 0
# print "len: %i" % len(result)
return struct.pack("%iB" % len(result), *result)
FIELDS = [
"m_am_vibrato_eg",
"c_am_vibrato_eg",
"m_ksl_volume",
"c_ksl_volume",
"m_attack_decay",
"c_attack_decay",
"m_sustain_release",
"c_sustain_release",
"m_waveform",
"c_waveform",
"feedback_fm",
"panning",
"finetune",
"voice_type",
]
def decode_type_9(data):
compressed_len, = struct.unpack("<H", data[0:2])
compressed_data = data[2 : 2 + compressed_len]
decompressed_data = decompress(compressed_data, compressed_len)
instr_data = {}
# Decode main fields:
for i in range(len(FIELDS)):
instr_data[FIELDS[i]], = struct.unpack(
"B", decompressed_data[i : i + 1]
)
# Decode instrument name
ps = decompressed_data[14:]
instr_name, = struct.unpack("%ip" % len(ps), ps)
instr_data["name"] = instr_name.decode("ascii")
return instr_data
def read(filename):
with open(filename, "rb") as f:
data = f.read()
hdrstr, crc, filever = struct.unpack("<7sHB", data[0:10])
hdrstr = hdrstr.decode("ascii")
if hdrstr.lower() != HEADER_STRING.lower():
raise Exception("Wrong file header ID string")
# TODO: CRC?
if filever == 9:
return decode_type_9(data[10:])
else:
raise Exception("Unsupported file version: %i" % filever)
if __name__ == "__main__":
for filename in sys.argv[1:]:
print(filename)
print(read(filename))