39 "/usr/share/soundfonts/default.sf2",
43 "/usr/share/sounds/sf3/default-GM.sf3",
46 "/usr/share/sounds/sf2/FluidR3_GM.sf2",
49 "/usr/share/soundfonts/FluidR3_GM.sf2",
52 "/usr/share/sounds/sf2/TimGM6mb.sf2",
53 "/usr/share/sounds/sf2/FluidR3_GS.sf2",
56static void RenderMusicStream(int16_t *buffer,
size_t samples)
58 std::unique_lock<std::mutex>
lock{
_midi.synth_mutex, std::try_to_lock};
60 if (!
lock.owns_lock() ||
_midi.synth ==
nullptr ||
_midi.player ==
nullptr)
return;
61 fluid_synth_write_s16(
_midi.synth, samples, buffer, 0, 2, buffer, 1, 2);
64static void load_and_execute_config_file(fluid_cmd_handler_t *cmd_handler,
const char *config_file)
66 if (std::filesystem::exists(config_file)) {
67 Debug(driver, 2,
"Fluidsynth: Attempting to load config file '{}'", config_file);
68 if (fluid_source(cmd_handler, config_file) < 0) {
69 Debug(driver, 0,
"Fluidsynth: Failed to execute command configuration file '{}'", config_file);
72 Debug(driver, 1,
"Fluidsynth: Failed to load config file '{}' - file doesn't exist", config_file);
78 std::lock_guard<std::mutex>
lock{
_midi.synth_mutex};
83 Debug(driver, 1,
"Fluidsynth: sf {}", sfont_name.has_value() ? *sfont_name :
"(null)");
86 _midi.settings = new_fluid_settings();
87 if (
_midi.settings ==
nullptr)
return "Could not create midi settings";
90 fluid_cmd_handler_t *cmd_handler = new_fluid_cmd_handler2(
_midi.settings,
nullptr,
nullptr,
nullptr);
91 if (cmd_handler == NULL)
return "Failed to create the early command handler";
93 char buf[MAX_PATH] = {};
94 const char *config_file = fluid_get_sysconf(buf,
sizeof(buf));
95 load_and_execute_config_file(cmd_handler, config_file);
96 config_file = fluid_get_userconf(buf,
sizeof(buf));
97 load_and_execute_config_file(cmd_handler, config_file);
99 delete_fluid_cmd_handler(cmd_handler);
102 fluid_settings_setint(
_midi.settings,
"synth.lock-memory", 0);
106 fluid_settings_setnum(
_midi.settings,
"synth.sample-rate", samplerate);
107 Debug(driver, 1,
"Fluidsynth: samplerate {:.0f}", (
float)samplerate);
110 _midi.synth = new_fluid_synth(
_midi.settings);
111 if (
_midi.synth ==
nullptr)
return "Could not open synth";
115 if (!sfont_name.has_value()) {
116 sfont_id = FLUID_FAILED;
119 char *default_soundfont =
nullptr;
120 fluid_settings_dupstr(
_midi.settings,
"synth.default-soundfont", &default_soundfont);
121 if (default_soundfont !=
nullptr && std::filesystem::exists(default_soundfont) && fluid_is_soundfont(default_soundfont)) {
122 sfont_id = fluid_synth_sfload(
_midi.synth, default_soundfont, 1);
124 fluid_free(default_soundfont);
127 if (sfont_id == FLUID_FAILED) {
129 if (!std::filesystem::exists(soundfont) || !fluid_is_soundfont(soundfont))
continue;
130 sfont_id = fluid_synth_sfload(
_midi.synth, soundfont, 1);
131 if (sfont_id != FLUID_FAILED)
break;
134 if (sfont_id == FLUID_FAILED)
return "Could not open any sound font";
136 std::string name{sfont_name.value()};
137 sfont_id = fluid_synth_sfload(
_midi.synth, name.c_str(), 1);
138 if (sfont_id == FLUID_FAILED)
return "Could not open sound font";
142 if (fluid_settings_getnum(
_midi.settings,
"synth.gain", &
_midi.max_gain) != FLUID_OK) {
143 if (fluid_settings_getnum_default(
_midi.settings,
"synth.gain", &
_midi.max_gain) != FLUID_OK) {
145 _midi.max_gain = 0.2;
149 _midi.player =
nullptr;
158 std::lock_guard<std::mutex>
lock{
_midi.synth_mutex};
160 if (
_midi.player !=
nullptr) delete_fluid_player(
_midi.player);
161 _midi.player =
nullptr;
163 if (
_midi.synth !=
nullptr) delete_fluid_synth(
_midi.synth);
164 _midi.synth =
nullptr;
166 if (
_midi.settings !=
nullptr) delete_fluid_settings(
_midi.settings);
167 _midi.settings =
nullptr;
176 if (filename.empty()) {
180 std::lock_guard<std::mutex>
lock{
_midi.synth_mutex};
183 if (
_midi.player ==
nullptr) {
184 Debug(driver, 0,
"Could not create midi player");
188 if (fluid_player_add(
_midi.player, filename.c_str()) != FLUID_OK) {
189 Debug(driver, 0,
"Could not open music file");
190 delete_fluid_player(
_midi.player);
191 _midi.player =
nullptr;
194 if (fluid_player_play(
_midi.player) != FLUID_OK) {
195 Debug(driver, 0,
"Could not start midi player");
196 delete_fluid_player(
_midi.player);
197 _midi.player =
nullptr;
204 std::lock_guard<std::mutex>
lock{
_midi.synth_mutex};
206 if (
_midi.player ==
nullptr)
return;
208 fluid_player_stop(
_midi.player);
210 delete_fluid_player(
_midi.player);
211 fluid_synth_system_reset(
_midi.synth);
212 fluid_synth_all_sounds_off(
_midi.synth, -1);
213 _midi.player =
nullptr;
218 std::lock_guard<std::mutex>
lock{
_midi.synth_mutex};
219 if (
_midi.player ==
nullptr)
return false;
221 return fluid_player_get_status(
_midi.player) == FLUID_PLAYER_PLAYING;
226 std::lock_guard<std::mutex>
lock{
_midi.synth_mutex};
227 if (
_midi.settings ==
nullptr)
return;
230 double gain = (1.0 * vol) / 128.0 *
_midi.max_gain;
231 if (fluid_settings_setnum(
_midi.settings,
"synth.gain", gain) != FLUID_OK) {
232 Debug(driver, 0,
"Could not set volume");
Factory for the fluidsynth driver.
std::optional< std::string_view > Start(const StringList ¶m) override
Start this driver.
void PlaySong(const MusicSongInfo &song) override
Play a particular song.
bool IsSongPlaying() override
Are we currently playing a song?
void StopSong() override
Stop playing the current song.
void SetVolume(uint8_t vol) override
Set the volume, if possible.
void Stop() override
Stop this driver.
Functions related to debugging.
#define Debug(category, level, format_string,...)
Output a line of debugging information.
std::optional< std::string_view > GetDriverParam(const StringList &parm, std::string_view name)
Get a string parameter the list of parameters.
std::mutex synth_mutex
Guard mutex for synth access.
static FMusicDriver_FluidSynth iFMusicDriver_FluidSynth
Factory for the FluidSynth driver.
static const char * _default_soundfonts[]
List of sound fonts to try by default.
fluid_player_t * player
FluidSynth MIDI player handle.
fluid_synth_t * synth
FluidSynth synthesizer handle.
double max_gain
Default max gain.
static struct @363362336073004235014321267016377305167030167165 _midi
Metadata about the midi we're playing.
fluid_settings_t * settings
FluidSynth settings handle.
Base for FluidSynth music playback.
Parser for standard MIDI files.
uint32_t MxSetMusicSource(MxStreamCallback music_callback)
Set source of PCM music.
Functions to mix sound samples.
A number of safeguards to prevent using unsafe methods.
Definition of base types and functions in a cross-platform compatible way.
std::vector< std::string > StringList
Type for a list of strings.
static std::string GetSMFFile(const MusicSongInfo &song)
Get the name of a Standard MIDI File for a given song.
Metadata about a music track.
std::mutex lock
synchronization for playback status fields