#include "th03/snd/snd.h" typedef enum { SND_BGM_OFF = 0, SND_BGM_FM26 = 1, SND_BGM_FM86 = 2, SND_BGM_MODE_COUNT = 3, SND_BGM_MIDI = 3, // unsupported _snd_bgm_mode_t_FORCE_UINT8 = 0xFF } snd_bgm_mode_t; typedef enum { SND_SE_OFF = 0, SND_SE_FM = 1, SND_SE_BEEP = 2, SND_SE_MODE_COUNT = 3, _snd_se_mode_t_FORCE_INT16 = 0x7FFF } snd_se_mode_t; extern unsigned char snd_se_mode; extern snd_bgm_mode_t snd_bgm_mode; #ifdef __cplusplus static inline bool snd_bgm_active() { return snd_bgm_mode; } static inline bool16 snd_se_active() { return (snd_se_mode != SND_SE_OFF); } #ifdef X86REAL_H // MODDERS: Just use [new_se] directly. static inline int16_t snd_get_param(int16_t ¶m) { _BX = _SP; return peek(_SS, (_BX + 4)); } #endif #endif #define snd_bgm_is_fm() \ (snd_bgm_mode != SND_BGM_MIDI) // Checks the requested BGM and SE modes against the available hardware and // sets [snd_se_mode] and [snd_bgm_mode] accordingly. Returns [snd_bgm_mode]. int pascal snd_determine_modes(int req_bgm_mode, int req_se_mode); #if defined(PMD) && defined(MASTER_HPP) // Loads a song ([func] == SND_LOAD_SONG) or a sound effect bank ([func] == // SND_LOAD_SE) into the respective work buffer of the sound driver. [fn] must // be null-terminated (despite the fixed length) and not have any extension. // Depending on [snd_bgm_mode], [snd_se_mode], and game, the following file is // loaded: // // | | TH04 | TH05 | // |----------------+------------+-----------| // | [snd_bgm_mode] | [func] = SND_LOAD_SONG | // |----------------+------------+-----------| // | SND_BGM_FM26 | [fn].m26 | [fn].m | // | SND_BGM_FM86 | [fn].m86 | [fn].m2 | // | SND_BGM_MIDI | [fn].mmd | [fn].md | (yes, see TH05's load[data].asm) // |----------------+------------+-----------| // | [snd_se_mode] | [func] = SND_LOAD_SE | // |----------------+------------+-----------| // | SND_SE_FM | [fn].efc | [fn].efc | // | SND_SE_BEEP | [fn].efs | [fn].efs | (using master.lib's BGM driver) // // The TH04/TH05 implementation stops any currently playing song before loading // a new one, avoiding the landmine from the TH02/TH03 version. // ZUN landmine: However, it adds three more: // • In SND_SE_BEEP mode, the TH04 version requires master.lib's BGM subsystem // to have been initialized before. TH05 initializes it on demand. // • The TH04 version does not handle file errors. // • The TH05 version will infinitely loop if neither the file for the current // [snd_bgm_mode] nor "[fn].m" exist. void pascal snd_load(const char fn[PF_FN_LEN], snd_load_func_t func); #if defined(__cplusplus) && (GAME == 5) // Refer to TH02's implementation for an explanation of how wrong this is. static inline uint16_t snd_load_size() { return 0xFFFF; } #endif #endif