Skip to content

Commit e1863ea

Browse files
committed
NSS: fix SSG playback with default Furnace SSG output volume
By default, Furnace uses two different output volumes for ADPCM/FM and SSG. Since there is no global output volume on hardware, emulate the lower SSG volume by decreasing all volume macro steps by 1 (it roughly divides the volume by 2). Make nsstool warn about this to give users a chance to compose their module with a full output for ADPCM/FM and SSG, so no tweak is necessary.
1 parent 14273fc commit e1863ea

File tree

1 file changed

+52
-3
lines changed

1 file changed

+52
-3
lines changed

tools/furtool.py

+52-3
Original file line numberDiff line numberDiff line change
@@ -144,7 +144,13 @@ class fur_module:
144144
samples: list[int] = field(default_factory=list)
145145

146146

147+
# Global tweak to compensate for default SSG output volume in Furnace
148+
HALF_SSG_VOL = False
149+
SSG_USED = False
150+
151+
147152
def read_module(bs):
153+
global HALF_SSG_VOL
148154
mod = fur_module()
149155
assert bs.read(16) == b"-Furnace module-" # magic
150156
bs.u2() # version
@@ -167,7 +173,15 @@ def read_module(bs):
167173
nb_patterns = bs.u4() # skip global pattern count
168174
chips = [x for x in bs.read(32)]
169175
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
171185
mod.name = bs.ustr()
172186
mod.author = bs.ustr()
173187
mod.pattern_len = pattern_len
@@ -219,6 +233,30 @@ def read_module(bs):
219233
return mod
220234

221235

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+
222260

223261
@dataclass
224262
class fm_operator:
@@ -420,11 +458,18 @@ def read_ssg_macro(length, bs):
420458
"arp": 1<<2, # BIT_LOAD_NOTE
421459
}
422460

423-
autoenv=False
461+
global SSG_USED
462+
SSG_USED = True
463+
autoenv = False
424464
blocks = {}
425465
macros, loop = read_macro_data(length, bs)
426466
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
428473

429474
# pass: merge waveform sequence into vol & noise_tone sequences for SSG registers
430475
if "wave" in blocks:
@@ -927,6 +972,10 @@ def main():
927972

928973
if arguments.action == "instruments":
929974
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.")
930979
elif arguments.action == "samples":
931980
generate_sample_map(m, smp, outfd)
932981
else:

0 commit comments

Comments
 (0)