@@ -144,7 +144,13 @@ class fur_module:
144
144
samples : list [int ] = field (default_factory = list )
145
145
146
146
147
+ # Global tweak to compensate for default SSG output volume in Furnace
148
+ HALF_SSG_VOL = False
149
+ SSG_USED = False
150
+
151
+
147
152
def read_module (bs ):
153
+ global HALF_SSG_VOL
148
154
mod = fur_module ()
149
155
assert bs .read (16 ) == b"-Furnace module-" # magic
150
156
bs .u2 () # version
@@ -167,7 +173,15 @@ def read_module(bs):
167
173
nb_patterns = bs .u4 () # skip global pattern count
168
174
chips = [x for x in bs .read (32 )]
169
175
assert chips [:chips .index (0 )] == [165 ] # single ym2610 chip
170
- bs .read (32 + 32 + 128 ) # skip chips vol, pan, flags
176
+ bs .read (32 ) # skip chips vol
177
+ bs .read (32 ) # skip chips pan
178
+ chip_flags = unpack ("32I" , bs .read (128 ))
179
+ # If the module has flags, check whether the master SSG and ADPCM/FM volumes match
180
+ # otherwise consider the module relies on Furnace defaults (currently half volume)
181
+ if chip_flags [0 ] != 0 :
182
+ check_chip_flags (chip_flags [0 ], bs )
183
+ else :
184
+ HALF_SSG_VOL = True
171
185
mod .name = bs .ustr ()
172
186
mod .author = bs .ustr ()
173
187
mod .pattern_len = pattern_len
@@ -219,6 +233,30 @@ def read_module(bs):
219
233
return mod
220
234
221
235
236
+ def check_chip_flags (ptr , bs ):
237
+ global HALF_SSG_VOL
238
+ saved_pos = bs .pos
239
+ bs .seek (ptr )
240
+ assert bs .read (4 ) == b"FLAG"
241
+ size = bs .u4 ()
242
+ data = bs .read (size )
243
+ datastr = bytearray (data ).decode ("utf-8" )
244
+ flags = {}
245
+ for d in datastr .split ("\n " )[:- 1 ]: # last item is "\0"
246
+ k , v = d .split ("=" )
247
+ flags [k ]= v
248
+ bs .seek (saved_pos )
249
+
250
+ # check if check volumes for SSG and FM/ADPCM (or the resp. Furnace defaults)
251
+ ssgVol = int (flags .get ("ssgVol" ,"128" ))
252
+ HALF_SSG_VOL = (ssgVol == 128 )
253
+
254
+ # warn if this moudle uses unsupported master volumes
255
+ fmVol = int (flags .get ("fmVol" ,"256" ))
256
+ if (not (fmVol == 256 and ssgVol == 128 )) and (not (fmVol == 256 and ssgVol == 256 )):
257
+ print ("Furnace module uses custom output volumes (FM/ADPCM: %d, SSG: %d), which is not supported in hardware.\n "
258
+ "Please set the volume of FM/ADPCM and SSG tracks to 256 in Furnace in the 'Chip Manager' window" % (fmVol , ssgVol ))
259
+
222
260
223
261
@dataclass
224
262
class fm_operator :
@@ -420,11 +458,18 @@ def read_ssg_macro(length, bs):
420
458
"arp" : 1 << 2 , # BIT_LOAD_NOTE
421
459
}
422
460
423
- autoenv = False
461
+ global SSG_USED
462
+ SSG_USED = True
463
+ autoenv = False
424
464
blocks = {}
425
465
macros , loop = read_macro_data (length , bs )
426
466
for code in macros :
427
- blocks [code_name [code ]] = macros [code ]
467
+ data = macros [code ]
468
+ if HALF_SSG_VOL and code_name [code ] == "vol" :
469
+ # Temporary volume adjustment when SSG channel is not mixed 100%
470
+ # in the furnace module
471
+ data = [max (0 , x - 1 ) for x in data ]
472
+ blocks [code_name [code ]] = data
428
473
429
474
# pass: merge waveform sequence into vol & noise_tone sequences for SSG registers
430
475
if "wave" in blocks :
@@ -927,6 +972,10 @@ def main():
927
972
928
973
if arguments .action == "instruments" :
929
974
generate_instruments (m , sample_map , name , bank , ins , outfd )
975
+ # warn if SSG required volume tweak
976
+ if SSG_USED and HALF_SSG_VOL :
977
+ print ("Volume of SSG channel is halved in Furnace, SSG macros were tweaked to compensate.\n "
978
+ "To fix that, please set the SSG volume to 256 in Furnace in the 'Chip Manager' window." )
930
979
elif arguments .action == "samples" :
931
980
generate_sample_map (m , smp , outfd )
932
981
else :
0 commit comments