RetroPlatformer/addons/modplayer/XM.gd

288 lines
7.7 KiB
GDScript3
Raw Normal View History

"""
XM reader by @arlez80
This script required Mod.gd for declaring the data structures.
MIT License
"""
class_name XM
enum XMNoteFlags {
NOTE = 0x01,
INSTRUMENT = 0x02,
VOLUME = 0x04,
EFFECT_COMMAND = 0x08,
EFFECT_PARAM = 0x10,
FIRST_BYTE_AS_FLAGS = 0x80,
}
"""
@param path
@return xm
"""
func read_file( path:String ) -> Mod.ModParseResult:
var f:File = File.new( )
var result: = Mod.ModParseResult.new( )
var err:int = f.open( path, f.READ )
if err != OK:
result.error = err
return result
var stream:StreamPeerBuffer = StreamPeerBuffer.new( )
stream.set_data_array( f.get_buffer( f.get_len( ) ) )
f.close( )
result.data = self._read( stream )
if result.data == null:
result.error = ERR_PARSE_ERROR
return result
"""
@param data
@return xm
"""
func read_data( data:PoolByteArray ) -> Mod.ModParseResult:
var stream:StreamPeerBuffer = StreamPeerBuffer.new( )
stream.set_data_array( data )
var result: = Mod.ModParseResult.new( )
result.data = self._read( stream )
if result.data == null:
result.error = ERR_PARSE_ERROR
return result
"""
@param stream
@return xm
"""
func _read( stream:StreamPeerBuffer ) -> Mod.ModData:
var xm: = Mod.ModData.new( )
if self._read_string( stream, 17 ) != "Extended Module: ":
return null
xm.module_name = self._read_string( stream, 20 )
if stream.get_u8( ) != 0x1A:
return null
xm.tracker_name = self._read_string( stream, 20 )
xm.version = stream.get_u16( )
var header_size:int = stream.get_u32( )
xm.song_length = stream.get_u16( )
xm.restart_position = stream.get_u16( )
xm.channel_count = stream.get_u16( )
var pattern_count:int = stream.get_u16( )
var instrument_count:int = stream.get_u16( )
xm.flags = stream.get_u16( )
xm.init_tick = stream.get_u16( )
xm.init_bpm = stream.get_u16( )
xm.song_positions = stream.get_partial_data( header_size - 20 )[1]
xm.patterns = self._read_patterns( stream, xm.flags, pattern_count, xm.channel_count )
if len( xm.patterns ) == 0:
return null
xm.instruments = self._read_instruments( stream, instrument_count )
if len( xm.instruments ) == 0:
return null
return xm
"""
@param stream
@param xm_file_flags XMデータに関するフラグ
@param pattern_count
@param channles
@return
"""
func _read_patterns( stream:StreamPeerBuffer, xm_file_flags:int, pattern_count:int, channels:int ) -> Array:
var patterns:Array = []
for i in range( pattern_count ):
var pattern:Array = []
if stream.get_u32( ) != 9:
return []
if stream.get_u8( ) != 0:
return []
var row_count:int = stream.get_u16( )
var size:int = stream.get_u16( )
for k in range( row_count ):
var line:Array = []
for ch in range( channels ):
var patr: = Mod.ModPatternNote.new( )
var first:int = stream.get_u8( )
if first & XMNoteFlags.FIRST_BYTE_AS_FLAGS != 0:
if first & XMNoteFlags.NOTE != 0:
patr.note = stream.get_u8( )
else:
patr.note = first
first = 0xFF
if 0 < patr.note:
patr.note -= 1
if first & XMNoteFlags.INSTRUMENT != 0:
patr.instrument = stream.get_u8( )
if first & XMNoteFlags.VOLUME != 0:
patr.volume = stream.get_u8( )
if first & XMNoteFlags.EFFECT_COMMAND != 0:
patr.effect_command = stream.get_u8( )
if first & XMNoteFlags.EFFECT_PARAM != 0:
patr.effect_param = stream.get_u8( )
if 0 < patr.note:
if xm_file_flags & Mod.ModFlags.LINEAR_FREQUENCY_TABLE != 0:
patr.key_number = self._conv_linear_freq( patr.note )
else:
patr.key_number = self._conv_amiga_freq( patr.note )
line.append( patr )
pattern.append( line )
patterns.append( pattern )
return patterns
"""
AMIGA式で周波数計算する
@param note
@return
"""
func _conv_amiga_freq( note:int ) -> int:
return int( 6848.0 / pow( 2.0, note / 12.0 ) )
"""
@param note
@return
"""
func _conv_linear_freq( note:int ) -> int:
return 7680 - note * 64
"""
@param stream
@param instrument_count
@return
"""
func _read_instruments( stream:StreamPeerBuffer, instrument_count:int ) -> Array:
var instruments:Array = []
for i in range( instrument_count ):
var inst: = Mod.ModInstrument.new( )
var size:int = stream.get_u32( )
var remain_size:int = size
inst.name = self._read_string( stream, 22 )
#print( "%d/%d %s" % [ i, instrument_count, inst.name] )
if stream.get_u8( ) != 0:
return []
var sample_count:int = stream.get_u16( )
var sample_numbers:Array = []
remain_size -= 4 + 22 + 1 + 2
if 0 < sample_count:
var sample_header_size:int = stream.get_u32( )
sample_numbers = stream.get_partial_data( 96 )[1]
inst.volume_envelope = Mod.ModEnvelope.new( )
for k in range( 12 ):
var ve: = Mod.ModEnvelopePoint.new( )
ve.frame = stream.get_u16( )
ve.value = stream.get_u16( )
inst.volume_envelope.points.append( ve )
inst.panning_envelope = Mod.ModEnvelope.new( )
for k in range( 12 ):
var pe: = Mod.ModEnvelopePoint.new( )
pe.frame = stream.get_u16( )
pe.value = stream.get_u16( )
inst.panning_envelope.points.append( pe )
remain_size -= 4 + 96 + 48 + 48
inst.volume_envelope.point_count = stream.get_u8( )
inst.panning_envelope.point_count = stream.get_u8( )
inst.volume_envelope.sustain_point = stream.get_u8( )
inst.volume_envelope.loop_start_point = stream.get_u8( )
inst.volume_envelope.loop_end_point = stream.get_u8( )
inst.panning_envelope.sustain_point = stream.get_u8( )
inst.panning_envelope.loop_start_point = stream.get_u8( )
inst.panning_envelope.loop_end_point = stream.get_u8( )
inst.volume_envelope.set_flag( stream.get_u8( ) )
inst.panning_envelope.set_flag( stream.get_u8( ) )
inst.vibrato_type = stream.get_u8( )
inst.vibrato_speed = stream.get_u8( )
inst.vibrato_depth = stream.get_u8( )
inst.vibrato_depth_shift = stream.get_u8( )
inst.volume_fadeout = stream.get_u16( )
remain_size -= 16
if 0 < remain_size:
stream.get_partial_data( remain_size ) # reserved
if 0 < sample_count:
var sounds:Array = []
# Sound Header
for k in range( sample_count ):
var xms: = Mod.ModSample.new( )
xms.length = stream.get_u32( )
xms.loop_start = stream.get_u32( )
xms.loop_length = stream.get_u32( )
xms.volume = stream.get_u8( )
xms.finetune = stream.get_8( )
xms.loop_type = stream.get_u8( )
if xms.loop_type & 16 != 0:
xms.bit = 16
xms.loop_type &= 3
xms.panning = stream.get_u8( )
xms.relative_note = stream.get_8( )
stream.get_u8( ) # Reserved
xms.name = self._read_string( stream, 22 )
sounds.append( xms )
# Sound Data
for xms in sounds:
var d:Array = []
var p:int = 0
if xms.bit == 16:
for k in range( xms.length >> 1 ):
p += stream.get_16( )
d.append( p & 0xFF )
d.append( ( p >> 8 ) & 0xFF )
else:
for k in range( xms.length ):
p += stream.get_8( )
d.append( p & 0xFF )
xms.data = PoolByteArray( d )
inst.samples = []
for k in sample_numbers:
if len( sounds ) <= k:
inst.samples.append( sounds[0] )
else:
inst.samples.append( sounds[k] )
instruments.append( inst )
return instruments
"""
@param stream
@param size
@return
"""
func _read_string( stream:StreamPeerBuffer, size:int ) -> String:
return stream.get_partial_data( size )[1].get_string_from_ascii( )