Audio plugin host https://kx.studio/carla
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

FluidSynthPlugin.cpp 64KB

11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756
  1. /*
  2. * Carla FluidSynth Plugin
  3. * Copyright (C) 2011-2014 Filipe Coelho <falktx@falktx.com>
  4. *
  5. * This program is free software; you can redistribute it and/or
  6. * modify it under the terms of the GNU General Public License as
  7. * published by the Free Software Foundation; either version 2 of
  8. * the License, or any later version.
  9. *
  10. * This program is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU General Public License for more details.
  14. *
  15. * For a full copy of the GNU General Public License see the doc/GPL.txt file.
  16. */
  17. #include "CarlaPluginInternal.hpp"
  18. #include "CarlaEngine.hpp"
  19. #ifdef WANT_FLUIDSYNTH
  20. #include "CarlaMathUtils.hpp"
  21. #include <fluidsynth.h>
  22. #include <QtCore/QStringList>
  23. #if (FLUIDSYNTH_VERSION_MAJOR >= 1 && FLUIDSYNTH_VERSION_MINOR >= 1 && FLUIDSYNTH_VERSION_MICRO >= 4)
  24. # define FLUIDSYNTH_VERSION_NEW_API
  25. #endif
  26. CARLA_BACKEND_START_NAMESPACE
  27. #if 0
  28. }
  29. #endif
  30. #define FLUID_DEFAULT_POLYPHONY 64
  31. class FluidSynthPlugin : public CarlaPlugin
  32. {
  33. public:
  34. FluidSynthPlugin(CarlaEngine* const engine, const unsigned int id, const bool use16Outs)
  35. : CarlaPlugin(engine, id),
  36. fUses16Outs(use16Outs),
  37. fSettings(nullptr),
  38. fSynth(nullptr),
  39. fSynthId(0),
  40. fAudio16Buffers(nullptr),
  41. fLabel(nullptr)
  42. {
  43. carla_debug("FluidSynthPlugin::FluidSynthPlugin(%p, %i, %s)", engine, id, bool2str(use16Outs));
  44. FLOAT_CLEAR(fParamBuffers, FluidSynthParametersMax);
  45. carla_fill<int32_t>(fCurMidiProgs, MAX_MIDI_CHANNELS, 0);
  46. // create settings
  47. fSettings = new_fluid_settings();
  48. CARLA_SAFE_ASSERT_RETURN(fSettings != nullptr,);
  49. // define settings
  50. fluid_settings_setint(fSettings, "synth.audio-channels", use16Outs ? 16 : 1);
  51. fluid_settings_setint(fSettings, "synth.audio-groups", use16Outs ? 16 : 1);
  52. fluid_settings_setnum(fSettings, "synth.sample-rate", pData->engine->getSampleRate());
  53. //fluid_settings_setnum(fSettings, "synth.cpu-cores", 2);
  54. fluid_settings_setint(fSettings, "synth.parallel-render", 1);
  55. fluid_settings_setint(fSettings, "synth.threadsafe-api", 0);
  56. // create synth
  57. fSynth = new_fluid_synth(fSettings);
  58. CARLA_SAFE_ASSERT_RETURN(fSynth != nullptr,);
  59. #ifdef FLUIDSYNTH_VERSION_NEW_API
  60. fluid_synth_set_sample_rate(fSynth, (float)pData->engine->getSampleRate());
  61. #endif
  62. // set default values
  63. fluid_synth_set_reverb_on(fSynth, 1);
  64. fluid_synth_set_reverb(fSynth, FLUID_REVERB_DEFAULT_ROOMSIZE, FLUID_REVERB_DEFAULT_DAMP, FLUID_REVERB_DEFAULT_WIDTH, FLUID_REVERB_DEFAULT_LEVEL);
  65. fluid_synth_set_chorus_on(fSynth, 1);
  66. fluid_synth_set_chorus(fSynth, FLUID_CHORUS_DEFAULT_N, FLUID_CHORUS_DEFAULT_LEVEL, FLUID_CHORUS_DEFAULT_SPEED, FLUID_CHORUS_DEFAULT_DEPTH, FLUID_CHORUS_DEFAULT_TYPE);
  67. fluid_synth_set_polyphony(fSynth, FLUID_DEFAULT_POLYPHONY);
  68. fluid_synth_set_gain(fSynth, 1.0f);
  69. for (int i=0; i < MAX_MIDI_CHANNELS; ++i)
  70. fluid_synth_set_interp_method(fSynth, i, FLUID_INTERP_DEFAULT);
  71. }
  72. ~FluidSynthPlugin() override
  73. {
  74. carla_debug("FluidSynthPlugin::~FluidSynthPlugin()");
  75. pData->singleMutex.lock();
  76. pData->masterMutex.lock();
  77. if (pData->client != nullptr && pData->client->isActive())
  78. pData->client->deactivate();
  79. if (pData->active)
  80. {
  81. deactivate();
  82. pData->active = false;
  83. }
  84. if (fSynth != nullptr)
  85. {
  86. delete_fluid_synth(fSynth);
  87. fSynth = nullptr;
  88. }
  89. if (fSettings != nullptr)
  90. {
  91. delete_fluid_settings(fSettings);
  92. fSettings = nullptr;
  93. }
  94. if (fLabel != nullptr)
  95. {
  96. delete[] fLabel;
  97. fLabel = nullptr;
  98. }
  99. clearBuffers();
  100. }
  101. // -------------------------------------------------------------------
  102. // Information (base)
  103. PluginType getType() const noexcept override
  104. {
  105. return PLUGIN_FILE_SF2;
  106. }
  107. PluginCategory getCategory() const noexcept override
  108. {
  109. return PLUGIN_CATEGORY_SYNTH;
  110. }
  111. // -------------------------------------------------------------------
  112. // Information (count)
  113. uint32_t getParameterScalePointCount(const uint32_t parameterId) const noexcept override
  114. {
  115. CARLA_SAFE_ASSERT_RETURN(parameterId < pData->param.count, 0);
  116. switch (parameterId)
  117. {
  118. case FluidSynthChorusType:
  119. return 2;
  120. case FluidSynthInterpolation:
  121. return 4;
  122. default:
  123. return 0;
  124. }
  125. }
  126. // -------------------------------------------------------------------
  127. // Information (current data)
  128. // nothing
  129. // -------------------------------------------------------------------
  130. // Information (per-plugin data)
  131. unsigned int getOptionsAvailable() const noexcept override
  132. {
  133. unsigned int options = 0x0;
  134. options |= PLUGIN_OPTION_MAP_PROGRAM_CHANGES;
  135. options |= PLUGIN_OPTION_SEND_CONTROL_CHANGES;
  136. options |= PLUGIN_OPTION_SEND_CHANNEL_PRESSURE;
  137. options |= PLUGIN_OPTION_SEND_PITCHBEND;
  138. options |= PLUGIN_OPTION_SEND_ALL_SOUND_OFF;
  139. return options;
  140. }
  141. float getParameterValue(const uint32_t parameterId) const noexcept override
  142. {
  143. CARLA_SAFE_ASSERT_RETURN(parameterId < pData->param.count, 0.0f);
  144. return fParamBuffers[parameterId];
  145. }
  146. float getParameterScalePointValue(const uint32_t parameterId, const uint32_t scalePointId) const noexcept override
  147. {
  148. CARLA_SAFE_ASSERT_RETURN(parameterId < pData->param.count, 0.0f);
  149. CARLA_SAFE_ASSERT_RETURN(scalePointId < getParameterScalePointCount(parameterId), 0.0f);
  150. switch (parameterId)
  151. {
  152. case FluidSynthChorusType:
  153. switch (scalePointId)
  154. {
  155. case 0:
  156. return FLUID_CHORUS_MOD_SINE;
  157. case 1:
  158. return FLUID_CHORUS_MOD_TRIANGLE;
  159. default:
  160. return FLUID_CHORUS_DEFAULT_TYPE;
  161. }
  162. case FluidSynthInterpolation:
  163. switch (scalePointId)
  164. {
  165. case 0:
  166. return FLUID_INTERP_NONE;
  167. case 1:
  168. return FLUID_INTERP_LINEAR;
  169. case 2:
  170. return FLUID_INTERP_4THORDER;
  171. case 3:
  172. return FLUID_INTERP_7THORDER;
  173. default:
  174. return FLUID_INTERP_DEFAULT;
  175. }
  176. default:
  177. return 0.0f;
  178. }
  179. }
  180. void getLabel(char* const strBuf) const noexcept override
  181. {
  182. if (fLabel != nullptr)
  183. std::strncpy(strBuf, fLabel, STR_MAX);
  184. else
  185. CarlaPlugin::getLabel(strBuf);
  186. }
  187. void getMaker(char* const strBuf) const noexcept override
  188. {
  189. std::strncpy(strBuf, "FluidSynth SF2 engine", STR_MAX);
  190. }
  191. void getCopyright(char* const strBuf) const noexcept override
  192. {
  193. std::strncpy(strBuf, "GNU GPL v2+", STR_MAX);
  194. }
  195. void getRealName(char* const strBuf) const noexcept override
  196. {
  197. getLabel(strBuf);
  198. }
  199. void getParameterName(const uint32_t parameterId, char* const strBuf) const noexcept override
  200. {
  201. CARLA_SAFE_ASSERT_RETURN(parameterId < pData->param.count,);
  202. switch (parameterId)
  203. {
  204. case FluidSynthReverbOnOff:
  205. std::strncpy(strBuf, "Reverb On/Off", STR_MAX);
  206. break;
  207. case FluidSynthReverbRoomSize:
  208. std::strncpy(strBuf, "Reverb Room Size", STR_MAX);
  209. break;
  210. case FluidSynthReverbDamp:
  211. std::strncpy(strBuf, "Reverb Damp", STR_MAX);
  212. break;
  213. case FluidSynthReverbLevel:
  214. std::strncpy(strBuf, "Reverb Level", STR_MAX);
  215. break;
  216. case FluidSynthReverbWidth:
  217. std::strncpy(strBuf, "Reverb Width", STR_MAX);
  218. break;
  219. case FluidSynthChorusOnOff:
  220. std::strncpy(strBuf, "Chorus On/Off", STR_MAX);
  221. break;
  222. case FluidSynthChorusNr:
  223. std::strncpy(strBuf, "Chorus Voice Count", STR_MAX);
  224. break;
  225. case FluidSynthChorusLevel:
  226. std::strncpy(strBuf, "Chorus Level", STR_MAX);
  227. break;
  228. case FluidSynthChorusSpeedHz:
  229. std::strncpy(strBuf, "Chorus Speed", STR_MAX);
  230. break;
  231. case FluidSynthChorusDepthMs:
  232. std::strncpy(strBuf, "Chorus Depth", STR_MAX);
  233. break;
  234. case FluidSynthChorusType:
  235. std::strncpy(strBuf, "Chorus Type", STR_MAX);
  236. break;
  237. case FluidSynthPolyphony:
  238. std::strncpy(strBuf, "Polyphony", STR_MAX);
  239. break;
  240. case FluidSynthInterpolation:
  241. std::strncpy(strBuf, "Interpolation", STR_MAX);
  242. break;
  243. case FluidSynthVoiceCount:
  244. std::strncpy(strBuf, "Voice Count", STR_MAX);
  245. break;
  246. default:
  247. CarlaPlugin::getParameterName(parameterId, strBuf);
  248. break;
  249. }
  250. }
  251. void getParameterUnit(const uint32_t parameterId, char* const strBuf) const noexcept override
  252. {
  253. CARLA_SAFE_ASSERT_RETURN(parameterId < pData->param.count,);
  254. switch (parameterId)
  255. {
  256. case FluidSynthChorusSpeedHz:
  257. std::strncpy(strBuf, "Hz", STR_MAX);
  258. break;
  259. case FluidSynthChorusDepthMs:
  260. std::strncpy(strBuf, "ms", STR_MAX);
  261. break;
  262. default:
  263. CarlaPlugin::getParameterUnit(parameterId, strBuf);
  264. break;
  265. }
  266. }
  267. void getParameterScalePointLabel(const uint32_t parameterId, const uint32_t scalePointId, char* const strBuf) const noexcept override
  268. {
  269. CARLA_SAFE_ASSERT_RETURN(parameterId < pData->param.count,);
  270. CARLA_SAFE_ASSERT_RETURN(scalePointId < getParameterScalePointCount(parameterId),);
  271. switch (parameterId)
  272. {
  273. case FluidSynthChorusType:
  274. switch (scalePointId)
  275. {
  276. case 0:
  277. std::strncpy(strBuf, "Sine wave", STR_MAX);
  278. return;
  279. case 1:
  280. std::strncpy(strBuf, "Triangle wave", STR_MAX);
  281. return;
  282. }
  283. case FluidSynthInterpolation:
  284. switch (scalePointId)
  285. {
  286. case 0:
  287. std::strncpy(strBuf, "None", STR_MAX);
  288. return;
  289. case 1:
  290. std::strncpy(strBuf, "Straight-line", STR_MAX);
  291. return;
  292. case 2:
  293. std::strncpy(strBuf, "Fourth-order", STR_MAX);
  294. return;
  295. case 3:
  296. std::strncpy(strBuf, "Seventh-order", STR_MAX);
  297. return;
  298. }
  299. }
  300. CarlaPlugin::getParameterScalePointLabel(parameterId, scalePointId, strBuf);
  301. }
  302. // -------------------------------------------------------------------
  303. // Set data (state)
  304. void prepareForSave() override
  305. {
  306. char strBuf[STR_MAX+1];
  307. std::snprintf(strBuf, STR_MAX, "%i:%i:%i:%i:%i:%i:%i:%i:%i:%i:%i:%i:%i:%i:%i:%i", fCurMidiProgs[0], fCurMidiProgs[1], fCurMidiProgs[2], fCurMidiProgs[3],
  308. fCurMidiProgs[4], fCurMidiProgs[5], fCurMidiProgs[6], fCurMidiProgs[7],
  309. fCurMidiProgs[8], fCurMidiProgs[9], fCurMidiProgs[10], fCurMidiProgs[11],
  310. fCurMidiProgs[12], fCurMidiProgs[13], fCurMidiProgs[14], fCurMidiProgs[15]);
  311. CarlaPlugin::setCustomData(CUSTOM_DATA_TYPE_STRING, "midiPrograms", strBuf, false);
  312. }
  313. // -------------------------------------------------------------------
  314. // Set data (internal stuff)
  315. void setCtrlChannel(const int8_t channel, const bool sendOsc, const bool sendCallback) noexcept override
  316. {
  317. if (channel < MAX_MIDI_CHANNELS)
  318. pData->midiprog.current = fCurMidiProgs[channel];
  319. CarlaPlugin::setCtrlChannel(channel, sendOsc, sendCallback);
  320. }
  321. // -------------------------------------------------------------------
  322. // Set data (plugin-specific stuff)
  323. void setParameterValue(const uint32_t parameterId, const float value, const bool sendGui, const bool sendOsc, const bool sendCallback) noexcept override
  324. {
  325. CARLA_SAFE_ASSERT_RETURN(parameterId < pData->param.count,);
  326. const float fixedValue(pData->param.getFixedValue(parameterId, value));
  327. fParamBuffers[parameterId] = fixedValue;
  328. {
  329. const ScopedSingleProcessLocker spl(this, (sendGui || sendOsc || sendCallback));
  330. switch (parameterId)
  331. {
  332. case FluidSynthReverbOnOff:
  333. try {
  334. fluid_synth_set_reverb_on(fSynth, (fixedValue > 0.5f) ? 1 : 0);
  335. } catch(...) {}
  336. break;
  337. case FluidSynthReverbRoomSize:
  338. case FluidSynthReverbDamp:
  339. case FluidSynthReverbLevel:
  340. case FluidSynthReverbWidth:
  341. try {
  342. fluid_synth_set_reverb(fSynth, fParamBuffers[FluidSynthReverbRoomSize], fParamBuffers[FluidSynthReverbDamp], fParamBuffers[FluidSynthReverbWidth], fParamBuffers[FluidSynthReverbLevel]);
  343. } catch(...) {}
  344. break;
  345. case FluidSynthChorusOnOff:
  346. try {
  347. fluid_synth_set_chorus_on(fSynth, (value > 0.5f) ? 1 : 0);
  348. } catch(...) {}
  349. break;
  350. case FluidSynthChorusNr:
  351. case FluidSynthChorusLevel:
  352. case FluidSynthChorusSpeedHz:
  353. case FluidSynthChorusDepthMs:
  354. case FluidSynthChorusType:
  355. try {
  356. fluid_synth_set_chorus(fSynth, (int)fParamBuffers[FluidSynthChorusNr], fParamBuffers[FluidSynthChorusLevel], fParamBuffers[FluidSynthChorusSpeedHz], fParamBuffers[FluidSynthChorusDepthMs], (int)fParamBuffers[FluidSynthChorusType]);
  357. } catch(...) {}
  358. break;
  359. case FluidSynthPolyphony:
  360. try {
  361. fluid_synth_set_polyphony(fSynth, (int)value);
  362. } catch(...) {}
  363. break;
  364. case FluidSynthInterpolation:
  365. for (int i=0; i < MAX_MIDI_CHANNELS; ++i)
  366. {
  367. try {
  368. fluid_synth_set_interp_method(fSynth, i, (int)value);
  369. }
  370. catch(...) {
  371. break;
  372. }
  373. }
  374. break;
  375. default:
  376. break;
  377. }
  378. }
  379. CarlaPlugin::setParameterValue(parameterId, value, sendGui, sendOsc, sendCallback);
  380. }
  381. void setCustomData(const char* const type, const char* const key, const char* const value, const bool sendGui) override
  382. {
  383. CARLA_SAFE_ASSERT_RETURN(fSynth != nullptr,);
  384. CARLA_SAFE_ASSERT_RETURN(type != nullptr && type[0] != '\0',);
  385. CARLA_SAFE_ASSERT_RETURN(key != nullptr && key[0] != '\0',);
  386. CARLA_SAFE_ASSERT_RETURN(value != nullptr && value[0] != '\0',);
  387. carla_debug("FluidSynthPlugin::setCustomData(%s, \"%s\", \"%s\", %s)", type, key, value, bool2str(sendGui));
  388. if (std::strcmp(type, CUSTOM_DATA_TYPE_STRING) != 0)
  389. return carla_stderr2("FluidSynthPlugin::setCustomData(\"%s\", \"%s\", \"%s\", %s) - type is not string", type, key, value, bool2str(sendGui));
  390. if (std::strcmp(key, "midiPrograms") != 0)
  391. return carla_stderr2("FluidSynthPlugin::setCustomData(\"%s\", \"%s\", \"%s\", %s) - type is not string", type, key, value, bool2str(sendGui));
  392. if (fUses16Outs)
  393. {
  394. QStringList midiProgramList(QString(value).split(":", QString::SkipEmptyParts));
  395. if (midiProgramList.count() == MAX_MIDI_CHANNELS)
  396. {
  397. int i = 0;
  398. foreach (const QString& midiProg, midiProgramList)
  399. {
  400. CARLA_SAFE_ASSERT_BREAK(i < MAX_MIDI_CHANNELS);
  401. bool ok;
  402. int index = midiProg.toInt(&ok);
  403. if (ok && index >= 0 && index < static_cast<int>(pData->midiprog.count))
  404. {
  405. const uint32_t bank = pData->midiprog.data[index].bank;
  406. const uint32_t program = pData->midiprog.data[index].program;
  407. fluid_synth_program_select(fSynth, i, fSynthId, bank, program);
  408. fCurMidiProgs[i] = index;
  409. if (pData->ctrlChannel == static_cast<int32_t>(i))
  410. {
  411. pData->midiprog.current = index;
  412. pData->engine->callback(ENGINE_CALLBACK_MIDI_PROGRAM_CHANGED, pData->id, index, 0, 0.0f, nullptr);
  413. }
  414. }
  415. ++i;
  416. }
  417. CARLA_SAFE_ASSERT(i == MAX_MIDI_CHANNELS);
  418. }
  419. }
  420. CarlaPlugin::setCustomData(type, key, value, sendGui);
  421. }
  422. void setMidiProgram(const int32_t index, const bool sendGui, const bool sendOsc, const bool sendCallback) noexcept override
  423. {
  424. CARLA_SAFE_ASSERT_RETURN(fSynth != nullptr,);
  425. CARLA_SAFE_ASSERT_RETURN(index >= -1 && index < static_cast<int32_t>(pData->midiprog.count),);
  426. if (index >= 0 && pData->ctrlChannel >= 0 && pData->ctrlChannel < MAX_MIDI_CHANNELS)
  427. {
  428. const uint32_t bank = pData->midiprog.data[index].bank;
  429. const uint32_t program = pData->midiprog.data[index].program;
  430. //const ScopedSingleProcessLocker spl(this, (sendGui || sendOsc || sendCallback));
  431. try {
  432. fluid_synth_program_select(fSynth, pData->ctrlChannel, fSynthId, bank, program);
  433. } catch(...) {}
  434. fCurMidiProgs[pData->ctrlChannel] = index;
  435. }
  436. CarlaPlugin::setMidiProgram(index, sendGui, sendOsc, sendCallback);
  437. }
  438. // -------------------------------------------------------------------
  439. // Set ui stuff
  440. // nothing
  441. // -------------------------------------------------------------------
  442. // Plugin state
  443. void reload() override
  444. {
  445. CARLA_SAFE_ASSERT_RETURN(pData->engine != nullptr,);
  446. CARLA_SAFE_ASSERT_RETURN(fSynth != nullptr,);
  447. carla_debug("FluidSynthPlugin::reload() - start");
  448. const EngineProcessMode processMode(pData->engine->getProccessMode());
  449. // Safely disable plugin for reload
  450. const ScopedDisabler sd(this);
  451. if (pData->active)
  452. deactivate();
  453. clearBuffers();
  454. uint32_t aOuts, params;
  455. aOuts = fUses16Outs ? 32 : 2;
  456. params = FluidSynthParametersMax;
  457. pData->audioOut.createNew(aOuts);
  458. pData->param.createNew(params, false);
  459. const uint portNameSize(pData->engine->getMaxPortNameSize());
  460. CarlaString portName;
  461. // ---------------------------------------
  462. // Audio Outputs
  463. if (fUses16Outs)
  464. {
  465. for (uint32_t i=0; i < 32; ++i)
  466. {
  467. portName.clear();
  468. if (processMode == ENGINE_PROCESS_MODE_SINGLE_CLIENT)
  469. {
  470. portName = pData->name;
  471. portName += ":";
  472. }
  473. portName += "out-";
  474. if ((i+2)/2 < 9)
  475. portName += "0";
  476. portName += CarlaString((i+2)/2);
  477. if (i % 2 == 0)
  478. portName += "L";
  479. else
  480. portName += "R";
  481. portName.truncate(portNameSize);
  482. pData->audioOut.ports[i].port = (CarlaEngineAudioPort*)pData->client->addPort(kEnginePortTypeAudio, portName, false);
  483. pData->audioOut.ports[i].rindex = i;
  484. }
  485. fAudio16Buffers = new float*[aOuts];
  486. for (uint32_t i=0; i < aOuts; ++i)
  487. fAudio16Buffers[i] = nullptr;
  488. }
  489. else
  490. {
  491. // out-left
  492. portName.clear();
  493. if (processMode == ENGINE_PROCESS_MODE_SINGLE_CLIENT)
  494. {
  495. portName = pData->name;
  496. portName += ":";
  497. }
  498. portName += "out-left";
  499. portName.truncate(portNameSize);
  500. pData->audioOut.ports[0].port = (CarlaEngineAudioPort*)pData->client->addPort(kEnginePortTypeAudio, portName, false);
  501. pData->audioOut.ports[0].rindex = 0;
  502. // out-right
  503. portName.clear();
  504. if (processMode == ENGINE_PROCESS_MODE_SINGLE_CLIENT)
  505. {
  506. portName = pData->name;
  507. portName += ":";
  508. }
  509. portName += "out-right";
  510. portName.truncate(portNameSize);
  511. pData->audioOut.ports[1].port = (CarlaEngineAudioPort*)pData->client->addPort(kEnginePortTypeAudio, portName, false);
  512. pData->audioOut.ports[1].rindex = 1;
  513. }
  514. // ---------------------------------------
  515. // Event Input
  516. {
  517. portName.clear();
  518. if (processMode == ENGINE_PROCESS_MODE_SINGLE_CLIENT)
  519. {
  520. portName = pData->name;
  521. portName += ":";
  522. }
  523. portName += "events-in";
  524. portName.truncate(portNameSize);
  525. pData->event.portIn = (CarlaEngineEventPort*)pData->client->addPort(kEnginePortTypeEvent, portName, true);
  526. }
  527. // ---------------------------------------
  528. // Event Output
  529. {
  530. portName.clear();
  531. if (processMode == ENGINE_PROCESS_MODE_SINGLE_CLIENT)
  532. {
  533. portName = pData->name;
  534. portName += ":";
  535. }
  536. portName += "events-out";
  537. portName.truncate(portNameSize);
  538. pData->event.portOut = (CarlaEngineEventPort*)pData->client->addPort(kEnginePortTypeEvent, portName, false);
  539. }
  540. // ---------------------------------------
  541. // Parameters
  542. {
  543. int j;
  544. // ----------------------
  545. j = FluidSynthReverbOnOff;
  546. pData->param.data[j].type = PARAMETER_INPUT;
  547. pData->param.data[j].hints = PARAMETER_IS_ENABLED /*| PARAMETER_IS_AUTOMABLE*/ | PARAMETER_IS_BOOLEAN;
  548. pData->param.data[j].index = j;
  549. pData->param.data[j].rindex = j;
  550. pData->param.data[j].midiCC = -1;
  551. pData->param.data[j].midiChannel = 0;
  552. pData->param.ranges[j].min = 0.0f;
  553. pData->param.ranges[j].max = 1.0f;
  554. pData->param.ranges[j].def = 1.0f;
  555. pData->param.ranges[j].step = 1.0f;
  556. pData->param.ranges[j].stepSmall = 1.0f;
  557. pData->param.ranges[j].stepLarge = 1.0f;
  558. pData->param.special[j] = PARAMETER_SPECIAL_NULL;
  559. fParamBuffers[j] = pData->param.ranges[j].def;
  560. // ----------------------
  561. j = FluidSynthReverbRoomSize;
  562. pData->param.data[j].type = PARAMETER_INPUT;
  563. pData->param.data[j].hints = PARAMETER_IS_ENABLED /*| PARAMETER_IS_AUTOMABLE*/;
  564. pData->param.data[j].index = j;
  565. pData->param.data[j].rindex = j;
  566. pData->param.data[j].midiCC = -1;
  567. pData->param.data[j].midiChannel = 0;
  568. pData->param.ranges[j].min = 0.0f;
  569. pData->param.ranges[j].max = 1.2f;
  570. pData->param.ranges[j].def = FLUID_REVERB_DEFAULT_ROOMSIZE;
  571. pData->param.ranges[j].step = 0.01f;
  572. pData->param.ranges[j].stepSmall = 0.0001f;
  573. pData->param.ranges[j].stepLarge = 0.1f;
  574. pData->param.special[j] = PARAMETER_SPECIAL_NULL;
  575. fParamBuffers[j] = pData->param.ranges[j].def;
  576. // ----------------------
  577. j = FluidSynthReverbDamp;
  578. pData->param.data[j].type = PARAMETER_INPUT;
  579. pData->param.data[j].hints = PARAMETER_IS_ENABLED /*| PARAMETER_IS_AUTOMABLE*/;
  580. pData->param.data[j].index = j;
  581. pData->param.data[j].rindex = j;
  582. pData->param.data[j].midiCC = -1;
  583. pData->param.data[j].midiChannel = 0;
  584. pData->param.ranges[j].min = 0.0f;
  585. pData->param.ranges[j].max = 1.0f;
  586. pData->param.ranges[j].def = FLUID_REVERB_DEFAULT_DAMP;
  587. pData->param.ranges[j].step = 0.01f;
  588. pData->param.ranges[j].stepSmall = 0.0001f;
  589. pData->param.ranges[j].stepLarge = 0.1f;
  590. pData->param.special[j] = PARAMETER_SPECIAL_NULL;
  591. fParamBuffers[j] = pData->param.ranges[j].def;
  592. // ----------------------
  593. j = FluidSynthReverbLevel;
  594. pData->param.data[j].type = PARAMETER_INPUT;
  595. pData->param.data[j].hints = PARAMETER_IS_ENABLED /*| PARAMETER_IS_AUTOMABLE*/;
  596. pData->param.data[j].index = j;
  597. pData->param.data[j].rindex = j;
  598. pData->param.data[j].midiCC = MIDI_CONTROL_REVERB_SEND_LEVEL;
  599. pData->param.data[j].midiChannel = 0;
  600. pData->param.ranges[j].min = 0.0f;
  601. pData->param.ranges[j].max = 1.0f;
  602. pData->param.ranges[j].def = FLUID_REVERB_DEFAULT_LEVEL;
  603. pData->param.ranges[j].step = 0.01f;
  604. pData->param.ranges[j].stepSmall = 0.0001f;
  605. pData->param.ranges[j].stepLarge = 0.1f;
  606. pData->param.special[j] = PARAMETER_SPECIAL_NULL;
  607. fParamBuffers[j] = pData->param.ranges[j].def;
  608. // ----------------------
  609. j = FluidSynthReverbWidth;
  610. pData->param.data[j].type = PARAMETER_INPUT;
  611. pData->param.data[j].hints = PARAMETER_IS_ENABLED /*| PARAMETER_IS_AUTOMABLE*/;
  612. pData->param.data[j].index = j;
  613. pData->param.data[j].rindex = j;
  614. pData->param.data[j].midiCC = -1;
  615. pData->param.data[j].midiChannel = 0;
  616. pData->param.ranges[j].min = 0.0f;
  617. pData->param.ranges[j].max = 10.0f; // should be 100, but that sounds too much
  618. pData->param.ranges[j].def = FLUID_REVERB_DEFAULT_WIDTH;
  619. pData->param.ranges[j].step = 0.01f;
  620. pData->param.ranges[j].stepSmall = 0.0001f;
  621. pData->param.ranges[j].stepLarge = 0.1f;
  622. pData->param.special[j] = PARAMETER_SPECIAL_NULL;
  623. fParamBuffers[j] = pData->param.ranges[j].def;
  624. // ----------------------
  625. j = FluidSynthChorusOnOff;
  626. pData->param.data[j].type = PARAMETER_INPUT;
  627. pData->param.data[j].hints = PARAMETER_IS_ENABLED | PARAMETER_IS_BOOLEAN;
  628. pData->param.data[j].index = j;
  629. pData->param.data[j].rindex = j;
  630. pData->param.data[j].midiCC = -1;
  631. pData->param.data[j].midiChannel = 0;
  632. pData->param.ranges[j].min = 0.0f;
  633. pData->param.ranges[j].max = 1.0f;
  634. pData->param.ranges[j].def = 1.0f;
  635. pData->param.ranges[j].step = 1.0f;
  636. pData->param.ranges[j].stepSmall = 1.0f;
  637. pData->param.ranges[j].stepLarge = 1.0f;
  638. pData->param.special[j] = PARAMETER_SPECIAL_NULL;
  639. fParamBuffers[j] = pData->param.ranges[j].def;
  640. // ----------------------
  641. j = FluidSynthChorusNr;
  642. pData->param.data[j].type = PARAMETER_INPUT;
  643. pData->param.data[j].hints = PARAMETER_IS_ENABLED | PARAMETER_IS_INTEGER;
  644. pData->param.data[j].index = j;
  645. pData->param.data[j].rindex = j;
  646. pData->param.data[j].midiCC = -1;
  647. pData->param.data[j].midiChannel = 0;
  648. pData->param.ranges[j].min = 0.0f;
  649. pData->param.ranges[j].max = 99.0f;
  650. pData->param.ranges[j].def = FLUID_CHORUS_DEFAULT_N;
  651. pData->param.ranges[j].step = 1.0f;
  652. pData->param.ranges[j].stepSmall = 1.0f;
  653. pData->param.ranges[j].stepLarge = 10.0f;
  654. pData->param.special[j] = PARAMETER_SPECIAL_NULL;
  655. fParamBuffers[j] = pData->param.ranges[j].def;
  656. // ----------------------
  657. j = FluidSynthChorusLevel;
  658. pData->param.data[j].type = PARAMETER_INPUT;
  659. pData->param.data[j].hints = PARAMETER_IS_ENABLED;
  660. pData->param.data[j].index = j;
  661. pData->param.data[j].rindex = j;
  662. pData->param.data[j].midiCC = 0; //MIDI_CONTROL_CHORUS_SEND_LEVEL;
  663. pData->param.data[j].midiChannel = 0;
  664. pData->param.ranges[j].min = 0.0f;
  665. pData->param.ranges[j].max = 10.0f;
  666. pData->param.ranges[j].def = FLUID_CHORUS_DEFAULT_LEVEL;
  667. pData->param.ranges[j].step = 0.01f;
  668. pData->param.ranges[j].stepSmall = 0.0001f;
  669. pData->param.ranges[j].stepLarge = 0.1f;
  670. pData->param.special[j] = PARAMETER_SPECIAL_NULL;
  671. fParamBuffers[j] = pData->param.ranges[j].def;
  672. // ----------------------
  673. j = FluidSynthChorusSpeedHz;
  674. pData->param.data[j].type = PARAMETER_INPUT;
  675. pData->param.data[j].hints = PARAMETER_IS_ENABLED;
  676. pData->param.data[j].index = j;
  677. pData->param.data[j].rindex = j;
  678. pData->param.data[j].midiCC = -1;
  679. pData->param.data[j].midiChannel = 0;
  680. pData->param.ranges[j].min = 0.29f;
  681. pData->param.ranges[j].max = 5.0f;
  682. pData->param.ranges[j].def = FLUID_CHORUS_DEFAULT_SPEED;
  683. pData->param.ranges[j].step = 0.01f;
  684. pData->param.ranges[j].stepSmall = 0.0001f;
  685. pData->param.ranges[j].stepLarge = 0.1f;
  686. pData->param.special[j] = PARAMETER_SPECIAL_NULL;
  687. fParamBuffers[j] = pData->param.ranges[j].def;
  688. // ----------------------
  689. j = FluidSynthChorusDepthMs;
  690. pData->param.data[j].type = PARAMETER_INPUT;
  691. pData->param.data[j].hints = PARAMETER_IS_ENABLED;
  692. pData->param.data[j].index = j;
  693. pData->param.data[j].rindex = j;
  694. pData->param.data[j].midiCC = -1;
  695. pData->param.data[j].midiChannel = 0;
  696. pData->param.ranges[j].min = 0.0f;
  697. pData->param.ranges[j].max = float(2048.0 * 1000.0 / pData->engine->getSampleRate()); // FIXME?
  698. pData->param.ranges[j].def = FLUID_CHORUS_DEFAULT_DEPTH;
  699. pData->param.ranges[j].step = 0.01f;
  700. pData->param.ranges[j].stepSmall = 0.0001f;
  701. pData->param.ranges[j].stepLarge = 0.1f;
  702. pData->param.special[j] = PARAMETER_SPECIAL_NULL;
  703. fParamBuffers[j] = pData->param.ranges[j].def;
  704. // ----------------------
  705. j = FluidSynthChorusType;
  706. pData->param.data[j].type = PARAMETER_INPUT;
  707. pData->param.data[j].hints = PARAMETER_IS_ENABLED | PARAMETER_IS_INTEGER | PARAMETER_USES_SCALEPOINTS;
  708. pData->param.data[j].index = j;
  709. pData->param.data[j].rindex = j;
  710. pData->param.data[j].midiCC = -1;
  711. pData->param.data[j].midiChannel = 0;
  712. pData->param.ranges[j].min = FLUID_CHORUS_MOD_SINE;
  713. pData->param.ranges[j].max = FLUID_CHORUS_MOD_TRIANGLE;
  714. pData->param.ranges[j].def = FLUID_CHORUS_DEFAULT_TYPE;
  715. pData->param.ranges[j].step = 1.0f;
  716. pData->param.ranges[j].stepSmall = 1.0f;
  717. pData->param.ranges[j].stepLarge = 1.0f;
  718. pData->param.special[j] = PARAMETER_SPECIAL_NULL;
  719. fParamBuffers[j] = pData->param.ranges[j].def;
  720. // ----------------------
  721. j = FluidSynthPolyphony;
  722. pData->param.data[j].type = PARAMETER_INPUT;
  723. pData->param.data[j].hints = PARAMETER_IS_ENABLED | PARAMETER_IS_INTEGER;
  724. pData->param.data[j].index = j;
  725. pData->param.data[j].rindex = j;
  726. pData->param.data[j].midiCC = -1;
  727. pData->param.data[j].midiChannel = 0;
  728. pData->param.ranges[j].min = 1.0f;
  729. pData->param.ranges[j].max = 512.0f; // max theoric is 65535
  730. pData->param.ranges[j].def = (float)fluid_synth_get_polyphony(fSynth);
  731. pData->param.ranges[j].step = 1.0f;
  732. pData->param.ranges[j].stepSmall = 1.0f;
  733. pData->param.ranges[j].stepLarge = 10.0f;
  734. pData->param.special[j] = PARAMETER_SPECIAL_NULL;
  735. fParamBuffers[j] = pData->param.ranges[j].def;
  736. // ----------------------
  737. j = FluidSynthInterpolation;
  738. pData->param.data[j].type = PARAMETER_INPUT;
  739. pData->param.data[j].hints = PARAMETER_IS_ENABLED | PARAMETER_IS_INTEGER | PARAMETER_USES_SCALEPOINTS;
  740. pData->param.data[j].index = j;
  741. pData->param.data[j].rindex = j;
  742. pData->param.data[j].midiCC = -1;
  743. pData->param.data[j].midiChannel = 0;
  744. pData->param.ranges[j].min = FLUID_INTERP_NONE;
  745. pData->param.ranges[j].max = FLUID_INTERP_HIGHEST;
  746. pData->param.ranges[j].def = FLUID_INTERP_DEFAULT;
  747. pData->param.ranges[j].step = 1.0f;
  748. pData->param.ranges[j].stepSmall = 1.0f;
  749. pData->param.ranges[j].stepLarge = 1.0f;
  750. pData->param.special[j] = PARAMETER_SPECIAL_NULL;
  751. fParamBuffers[j] = pData->param.ranges[j].def;
  752. // ----------------------
  753. j = FluidSynthVoiceCount;
  754. pData->param.data[j].type = PARAMETER_OUTPUT;
  755. pData->param.data[j].hints = PARAMETER_IS_ENABLED | PARAMETER_IS_AUTOMABLE | PARAMETER_IS_INTEGER;
  756. pData->param.data[j].index = j;
  757. pData->param.data[j].rindex = j;
  758. pData->param.data[j].midiCC = -1;
  759. pData->param.data[j].midiChannel = 0;
  760. pData->param.ranges[j].min = 0.0f;
  761. pData->param.ranges[j].max = 65535.0f;
  762. pData->param.ranges[j].def = 0.0f;
  763. pData->param.ranges[j].step = 1.0f;
  764. pData->param.ranges[j].stepSmall = 1.0f;
  765. pData->param.ranges[j].stepLarge = 1.0f;
  766. pData->param.special[j] = PARAMETER_SPECIAL_NULL;
  767. fParamBuffers[j] = pData->param.ranges[j].def;
  768. }
  769. // ---------------------------------------
  770. // plugin hints
  771. pData->hints = 0x0;
  772. pData->hints |= PLUGIN_IS_SYNTH;
  773. pData->hints |= PLUGIN_CAN_VOLUME;
  774. if (! fUses16Outs)
  775. pData->hints |= PLUGIN_CAN_BALANCE;
  776. // extra plugin hints
  777. pData->extraHints = 0x0;
  778. pData->extraHints |= PLUGIN_EXTRA_HINT_HAS_MIDI_IN;
  779. if (! fUses16Outs)
  780. pData->extraHints |= PLUGIN_EXTRA_HINT_CAN_RUN_RACK;
  781. bufferSizeChanged(pData->engine->getBufferSize());
  782. reloadPrograms(true);
  783. if (pData->active)
  784. activate();
  785. carla_debug("FluidSynthPlugin::reload() - end");
  786. }
  787. void reloadPrograms(const bool doInit) override
  788. {
  789. carla_debug("FluidSynthPlugin::reloadPrograms(%s)", bool2str(doInit));
  790. // save drum info in case we have one program for it
  791. bool hasDrums = false;
  792. int32_t drumIndex;
  793. uint32_t drumProg;
  794. // Delete old programs
  795. pData->midiprog.clear();
  796. // Query new programs
  797. uint32_t count = 0;
  798. if (fluid_sfont_t* const f_sfont = fluid_synth_get_sfont_by_id(fSynth, fSynthId))
  799. {
  800. fluid_preset_t f_preset;
  801. // initial check to know how many midi-programs we have
  802. f_sfont->iteration_start(f_sfont);
  803. while (f_sfont->iteration_next(f_sfont, &f_preset))
  804. ++count;
  805. // sound kits must always have at least 1 midi-program
  806. CARLA_SAFE_ASSERT_RETURN(count > 0,);
  807. pData->midiprog.createNew(count);
  808. // Update data
  809. int tmp;
  810. uint32_t i = 0;
  811. f_sfont->iteration_start(f_sfont);
  812. while (f_sfont->iteration_next(f_sfont, &f_preset))
  813. {
  814. CARLA_SAFE_ASSERT_BREAK(i < count);
  815. tmp = f_preset.get_banknum(&f_preset);
  816. pData->midiprog.data[i].bank = (tmp >= 0) ? static_cast<uint32_t>(tmp) : 0;
  817. tmp = f_preset.get_num(&f_preset);
  818. pData->midiprog.data[i].program = (tmp >= 0) ? static_cast<uint32_t>(tmp) : 0;
  819. pData->midiprog.data[i].name = carla_strdup(f_preset.get_name(&f_preset));
  820. if (pData->midiprog.data[i].bank == 128 && ! hasDrums)
  821. {
  822. hasDrums = true;
  823. drumIndex = static_cast<int32_t>(i);
  824. drumProg = pData->midiprog.data[i].program;
  825. }
  826. ++i;
  827. }
  828. }
  829. else
  830. {
  831. // failing means 0 midi-programs, it shouldn't happen!
  832. carla_safe_assert("fluid_sfont_t* const f_sfont = fluid_synth_get_sfont_by_id(fSynth, fSynthId)", __FILE__, __LINE__);
  833. return;
  834. }
  835. #ifndef BUILD_BRIDGE
  836. // Update OSC Names
  837. if (pData->engine->isOscControlRegistered())
  838. {
  839. pData->engine->oscSend_control_set_midi_program_count(pData->id, count);
  840. for (uint32_t i=0; i < count; ++i)
  841. pData->engine->oscSend_control_set_midi_program_data(pData->id, i, pData->midiprog.data[i].bank, pData->midiprog.data[i].program, pData->midiprog.data[i].name);
  842. }
  843. #endif
  844. if (doInit)
  845. {
  846. fluid_synth_program_reset(fSynth);
  847. // select first program, or 128 for ch10
  848. for (int i=0; i < MAX_MIDI_CHANNELS && i != 9; ++i)
  849. {
  850. #ifdef FLUIDSYNTH_VERSION_NEW_API
  851. fluid_synth_set_channel_type(fSynth, i, CHANNEL_TYPE_MELODIC);
  852. #endif
  853. fluid_synth_program_select(fSynth, i, fSynthId, pData->midiprog.data[0].bank, pData->midiprog.data[0].program);
  854. fCurMidiProgs[i] = 0;
  855. }
  856. if (hasDrums)
  857. {
  858. #ifdef FLUIDSYNTH_VERSION_NEW_API
  859. fluid_synth_set_channel_type(fSynth, 9, CHANNEL_TYPE_DRUM);
  860. #endif
  861. fluid_synth_program_select(fSynth, 9, fSynthId, 128, drumProg);
  862. fCurMidiProgs[9] = drumIndex;
  863. }
  864. else
  865. {
  866. #ifdef FLUIDSYNTH_VERSION_NEW_API
  867. fluid_synth_set_channel_type(fSynth, 9, CHANNEL_TYPE_MELODIC);
  868. #endif
  869. fluid_synth_program_select(fSynth, 9, fSynthId, pData->midiprog.data[0].bank, pData->midiprog.data[0].program);
  870. fCurMidiProgs[9] = 0;
  871. }
  872. pData->midiprog.current = 0;
  873. }
  874. else
  875. {
  876. pData->engine->callback(ENGINE_CALLBACK_RELOAD_PROGRAMS, pData->id, 0, 0, 0.0f, nullptr);
  877. }
  878. }
  879. // -------------------------------------------------------------------
  880. // Plugin processing
  881. void process(float** const, float** const outBuffer, const uint32_t frames) override
  882. {
  883. // --------------------------------------------------------------------------------------------------------
  884. // Check if active
  885. if (! pData->active)
  886. {
  887. // disable any output sound
  888. for (uint32_t i=0; i < pData->audioOut.count; ++i)
  889. FLOAT_CLEAR(outBuffer[i], frames);
  890. return;
  891. }
  892. // --------------------------------------------------------------------------------------------------------
  893. // Check if needs reset
  894. if (pData->needsReset)
  895. {
  896. if (pData->options & PLUGIN_OPTION_SEND_ALL_SOUND_OFF)
  897. {
  898. for (int i=0; i < MAX_MIDI_CHANNELS; ++i)
  899. {
  900. #ifdef FLUIDSYNTH_VERSION_NEW_API
  901. fluid_synth_all_notes_off(fSynth, i);
  902. fluid_synth_all_sounds_off(fSynth, i);
  903. #else
  904. fluid_synth_cc(fSynth, i, MIDI_CONTROL_ALL_SOUND_OFF, 0);
  905. fluid_synth_cc(fSynth, i, MIDI_CONTROL_ALL_NOTES_OFF, 0);
  906. #endif
  907. }
  908. }
  909. else if (pData->ctrlChannel >= 0 && pData->ctrlChannel < MAX_MIDI_CHANNELS)
  910. {
  911. for (int i=0; i < MAX_MIDI_NOTE; ++i)
  912. fluid_synth_noteoff(fSynth, pData->ctrlChannel, i);
  913. }
  914. pData->needsReset = false;
  915. }
  916. // --------------------------------------------------------------------------------------------------------
  917. // Event Input and Processing
  918. {
  919. // ----------------------------------------------------------------------------------------------------
  920. // MIDI Input (External)
  921. if (pData->extNotes.mutex.tryLock())
  922. {
  923. while (! pData->extNotes.data.isEmpty())
  924. {
  925. const ExternalMidiNote& note(pData->extNotes.data.getFirst(true));
  926. CARLA_SAFE_ASSERT_CONTINUE(note.channel >= 0 && note.channel < MAX_MIDI_CHANNELS);
  927. if (note.velo > 0)
  928. fluid_synth_noteon(fSynth, note.channel, note.note, note.velo);
  929. else
  930. fluid_synth_noteoff(fSynth,note.channel, note.note);
  931. }
  932. pData->extNotes.mutex.unlock();
  933. } // End of MIDI Input (External)
  934. // ----------------------------------------------------------------------------------------------------
  935. // Event Input (System)
  936. bool allNotesOffSent = false;
  937. uint32_t time, nEvents = pData->event.portIn->getEventCount();
  938. uint32_t timeOffset = 0;
  939. uint32_t nextBankIds[MAX_MIDI_CHANNELS] = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 128, 0, 0, 0, 0, 0, 0 };
  940. if (pData->midiprog.current >= 0 && pData->midiprog.count > 0 && pData->ctrlChannel >= 0 && pData->ctrlChannel < MAX_MIDI_CHANNELS)
  941. nextBankIds[pData->ctrlChannel] = pData->midiprog.data[pData->midiprog.current].bank;
  942. for (uint32_t i=0; i < nEvents; ++i)
  943. {
  944. const EngineEvent& event(pData->event.portIn->getEvent(i));
  945. time = event.time;
  946. CARLA_SAFE_ASSERT_CONTINUE(time < frames);
  947. CARLA_SAFE_ASSERT_BREAK(time >= timeOffset);
  948. if (time > timeOffset)
  949. {
  950. if (processSingle(outBuffer, time - timeOffset, timeOffset))
  951. {
  952. timeOffset = time;
  953. if (pData->midiprog.current >= 0 && pData->midiprog.count > 0 && pData->ctrlChannel >= 0 && pData->ctrlChannel < MAX_MIDI_CHANNELS)
  954. nextBankIds[pData->ctrlChannel] = pData->midiprog.data[pData->midiprog.current].bank;
  955. }
  956. }
  957. // Control change
  958. switch (event.type)
  959. {
  960. case kEngineEventTypeNull:
  961. break;
  962. case kEngineEventTypeControl:
  963. {
  964. const EngineControlEvent& ctrlEvent = event.ctrl;
  965. switch (ctrlEvent.type)
  966. {
  967. case kEngineControlEventTypeNull:
  968. break;
  969. case kEngineControlEventTypeParameter:
  970. {
  971. #ifndef BUILD_BRIDGE
  972. // Control backend stuff
  973. if (event.channel == pData->ctrlChannel)
  974. {
  975. float value;
  976. if (MIDI_IS_CONTROL_BREATH_CONTROLLER(ctrlEvent.param) && (pData->hints & PLUGIN_CAN_DRYWET) != 0)
  977. {
  978. value = ctrlEvent.value;
  979. setDryWet(value, false, false);
  980. pData->postponeRtEvent(kPluginPostRtEventParameterChange, PARAMETER_DRYWET, 0, value);
  981. }
  982. if (MIDI_IS_CONTROL_CHANNEL_VOLUME(ctrlEvent.param) && (pData->hints & PLUGIN_CAN_VOLUME) != 0)
  983. {
  984. value = ctrlEvent.value*127.0f/100.0f;
  985. setVolume(value, false, false);
  986. pData->postponeRtEvent(kPluginPostRtEventParameterChange, PARAMETER_VOLUME, 0, value);
  987. }
  988. if (MIDI_IS_CONTROL_BALANCE(ctrlEvent.param) && (pData->hints & PLUGIN_CAN_BALANCE) != 0)
  989. {
  990. float left, right;
  991. value = ctrlEvent.value/0.5f - 1.0f;
  992. if (value < 0.0f)
  993. {
  994. left = -1.0f;
  995. right = (value*2.0f)+1.0f;
  996. }
  997. else if (value > 0.0f)
  998. {
  999. left = (value*2.0f)-1.0f;
  1000. right = 1.0f;
  1001. }
  1002. else
  1003. {
  1004. left = -1.0f;
  1005. right = 1.0f;
  1006. }
  1007. setBalanceLeft(left, false, false);
  1008. setBalanceRight(right, false, false);
  1009. pData->postponeRtEvent(kPluginPostRtEventParameterChange, PARAMETER_BALANCE_LEFT, 0, left);
  1010. pData->postponeRtEvent(kPluginPostRtEventParameterChange, PARAMETER_BALANCE_RIGHT, 0, right);
  1011. }
  1012. }
  1013. #endif
  1014. // Control plugin parameters
  1015. for (uint32_t k=0; k < pData->param.count; ++k)
  1016. {
  1017. if (pData->param.data[k].midiChannel != event.channel)
  1018. continue;
  1019. if (pData->param.data[k].midiCC != ctrlEvent.param)
  1020. continue;
  1021. if (pData->param.data[k].hints != PARAMETER_INPUT)
  1022. continue;
  1023. if ((pData->param.data[k].hints & PARAMETER_IS_AUTOMABLE) == 0)
  1024. continue;
  1025. float value;
  1026. if (pData->param.data[k].hints & PARAMETER_IS_BOOLEAN)
  1027. {
  1028. value = (ctrlEvent.value < 0.5f) ? pData->param.ranges[k].min : pData->param.ranges[k].max;
  1029. }
  1030. else
  1031. {
  1032. value = pData->param.ranges[k].getUnnormalizedValue(ctrlEvent.value);
  1033. if (pData->param.data[k].hints & PARAMETER_IS_INTEGER)
  1034. value = std::rint(value);
  1035. }
  1036. setParameterValue(k, value, false, false, false);
  1037. pData->postponeRtEvent(kPluginPostRtEventParameterChange, static_cast<int32_t>(k), 0, value);
  1038. }
  1039. if ((pData->options & PLUGIN_OPTION_SEND_CONTROL_CHANGES) != 0 && ctrlEvent.param <= 0x5F)
  1040. {
  1041. fluid_synth_cc(fSynth, event.channel, ctrlEvent.param, int(ctrlEvent.value*127.0f));
  1042. }
  1043. break;
  1044. }
  1045. case kEngineControlEventTypeMidiBank:
  1046. if (event.channel < MAX_MIDI_CHANNELS && (pData->options & PLUGIN_OPTION_MAP_PROGRAM_CHANGES) != 0)
  1047. nextBankIds[event.channel] = ctrlEvent.param;
  1048. break;
  1049. case kEngineControlEventTypeMidiProgram:
  1050. if (event.channel < MAX_MIDI_CHANNELS && (pData->options & PLUGIN_OPTION_MAP_PROGRAM_CHANGES) != 0)
  1051. {
  1052. const uint32_t bankId(nextBankIds[event.channel]);
  1053. const uint32_t progId(ctrlEvent.param);
  1054. for (uint32_t k=0; k < pData->midiprog.count; ++k)
  1055. {
  1056. if (pData->midiprog.data[k].bank == bankId && pData->midiprog.data[k].program == progId)
  1057. {
  1058. fluid_synth_program_select(fSynth, event.channel, fSynthId, bankId, progId);
  1059. fCurMidiProgs[event.channel] = static_cast<int32_t>(k);
  1060. if (event.channel == pData->ctrlChannel)
  1061. pData->postponeRtEvent(kPluginPostRtEventMidiProgramChange, static_cast<int32_t>(k), 0, 0.0f);
  1062. break;
  1063. }
  1064. }
  1065. }
  1066. break;
  1067. case kEngineControlEventTypeAllSoundOff:
  1068. if (pData->options & PLUGIN_OPTION_SEND_ALL_SOUND_OFF)
  1069. {
  1070. #ifdef FLUIDSYNTH_VERSION_NEW_API
  1071. fluid_synth_all_sounds_off(fSynth, event.channel);
  1072. #else
  1073. fluid_synth_cc(fSynth, event.channel, MIDI_CONTROL_ALL_SOUND_OFF, 0);
  1074. #endif
  1075. }
  1076. break;
  1077. case kEngineControlEventTypeAllNotesOff:
  1078. if (pData->options & PLUGIN_OPTION_SEND_ALL_SOUND_OFF)
  1079. {
  1080. if (event.channel == pData->ctrlChannel && ! allNotesOffSent)
  1081. {
  1082. allNotesOffSent = true;
  1083. sendMidiAllNotesOffToCallback();
  1084. }
  1085. #ifdef FLUIDSYNTH_VERSION_NEW_API
  1086. fluid_synth_all_notes_off(fSynth, event.channel);
  1087. #else
  1088. fluid_synth_cc(fSynth, event.channel, MIDI_CONTROL_ALL_NOTES_OFF, 0);
  1089. #endif
  1090. }
  1091. break;
  1092. }
  1093. break;
  1094. }
  1095. case kEngineEventTypeMidi:
  1096. {
  1097. const EngineMidiEvent& midiEvent(event.midi);
  1098. uint8_t status = uint8_t(MIDI_GET_STATUS_FROM_DATA(midiEvent.data));
  1099. uint8_t channel = event.channel;
  1100. // Fix bad note-off
  1101. if (MIDI_IS_STATUS_NOTE_ON(status) && midiEvent.data[2] == 0)
  1102. status = MIDI_STATUS_NOTE_OFF;
  1103. if (MIDI_IS_STATUS_NOTE_OFF(status))
  1104. {
  1105. const uint8_t note = midiEvent.data[1];
  1106. fluid_synth_noteoff(fSynth, channel, note);
  1107. pData->postponeRtEvent(kPluginPostRtEventNoteOff, channel, note, 0.0f);
  1108. }
  1109. else if (MIDI_IS_STATUS_NOTE_ON(status))
  1110. {
  1111. const uint8_t note = midiEvent.data[1];
  1112. const uint8_t velo = midiEvent.data[2];
  1113. fluid_synth_noteon(fSynth, channel, note, velo);
  1114. pData->postponeRtEvent(kPluginPostRtEventNoteOn, channel, note, velo);
  1115. }
  1116. else if (MIDI_IS_STATUS_POLYPHONIC_AFTERTOUCH(status) && (pData->options & PLUGIN_OPTION_SEND_NOTE_AFTERTOUCH) != 0)
  1117. {
  1118. //const uint8_t note = midiEvent.data[1];
  1119. //const uint8_t pressure = midiEvent.data[2];
  1120. // TODO, not in fluidsynth API
  1121. }
  1122. else if (MIDI_IS_STATUS_CONTROL_CHANGE(status) && (pData->options & PLUGIN_OPTION_SEND_CONTROL_CHANGES) != 0)
  1123. {
  1124. const uint8_t control = midiEvent.data[1];
  1125. const uint8_t value = midiEvent.data[2];
  1126. fluid_synth_cc(fSynth, channel, control, value);
  1127. }
  1128. else if (MIDI_IS_STATUS_CHANNEL_PRESSURE(status) && (pData->options & PLUGIN_OPTION_SEND_CHANNEL_PRESSURE) != 0)
  1129. {
  1130. const uint8_t pressure = midiEvent.data[1];
  1131. fluid_synth_channel_pressure(fSynth, channel, pressure);;
  1132. }
  1133. else if (MIDI_IS_STATUS_PITCH_WHEEL_CONTROL(status) && (pData->options & PLUGIN_OPTION_SEND_PITCHBEND) != 0)
  1134. {
  1135. const uint8_t lsb = midiEvent.data[1];
  1136. const uint8_t msb = midiEvent.data[2];
  1137. const int value = ((msb << 7) | lsb) - 8192;
  1138. fluid_synth_pitch_bend(fSynth, channel, value);
  1139. }
  1140. break;
  1141. }
  1142. }
  1143. }
  1144. pData->postRtEvents.trySplice();
  1145. if (frames > timeOffset)
  1146. processSingle(outBuffer, frames - timeOffset, timeOffset);
  1147. } // End of Event Input and Processing
  1148. CARLA_PROCESS_CONTINUE_CHECK;
  1149. // --------------------------------------------------------------------------------------------------------
  1150. // Control Output
  1151. {
  1152. uint32_t k = FluidSynthVoiceCount;
  1153. fParamBuffers[k] = float(fluid_synth_get_active_voice_count(fSynth));
  1154. pData->param.ranges[k].fixValue(fParamBuffers[k]);
  1155. if (pData->param.data[k].midiCC > 0)
  1156. {
  1157. float value(pData->param.ranges[k].getNormalizedValue(fParamBuffers[k]));
  1158. pData->event.portOut->writeControlEvent(0, pData->param.data[k].midiChannel, kEngineControlEventTypeParameter, static_cast<uint16_t>(pData->param.data[k].midiCC), value);
  1159. }
  1160. } // End of Control Output
  1161. }
  1162. bool processSingle(float** const outBuffer, const uint32_t frames, const uint32_t timeOffset)
  1163. {
  1164. CARLA_SAFE_ASSERT_RETURN(outBuffer != nullptr, false);
  1165. CARLA_SAFE_ASSERT_RETURN(frames > 0, false);
  1166. // --------------------------------------------------------------------------------------------------------
  1167. // Try lock, silence otherwise
  1168. if (pData->engine->isOffline())
  1169. {
  1170. pData->singleMutex.lock();
  1171. }
  1172. else if (! pData->singleMutex.tryLock())
  1173. {
  1174. for (uint32_t i=0; i < pData->audioOut.count; ++i)
  1175. {
  1176. for (uint32_t k=0; k < frames; ++k)
  1177. outBuffer[i][k+timeOffset] = 0.0f;
  1178. }
  1179. return false;
  1180. }
  1181. // --------------------------------------------------------------------------------------------------------
  1182. // Fill plugin buffers and Run plugin
  1183. if (fUses16Outs)
  1184. {
  1185. for (uint32_t i=0; i < pData->audioOut.count; ++i)
  1186. FLOAT_CLEAR(fAudio16Buffers[i], frames);
  1187. // FIXME use '32' or '16' instead of outs
  1188. fluid_synth_process(fSynth, static_cast<int>(frames), 0, nullptr, static_cast<int>(pData->audioOut.count), fAudio16Buffers);
  1189. }
  1190. else
  1191. fluid_synth_write_float(fSynth, static_cast<int>(frames), outBuffer[0] + timeOffset, 0, 1, outBuffer[1] + timeOffset, 0, 1);
  1192. #ifndef BUILD_BRIDGE
  1193. // --------------------------------------------------------------------------------------------------------
  1194. // Post-processing (volume and balance)
  1195. {
  1196. // note - balance not possible with fUses16Outs, so we can safely skip fAudioOutBuffers
  1197. const bool doVolume = (pData->hints & PLUGIN_CAN_VOLUME) > 0 && pData->postProc.volume != 1.0f;
  1198. const bool doBalance = (pData->hints & PLUGIN_CAN_BALANCE) > 0 && (pData->postProc.balanceLeft != -1.0f || pData->postProc.balanceRight != 1.0f);
  1199. float oldBufLeft[doBalance ? frames : 1];
  1200. for (uint32_t i=0; i < pData->audioOut.count; ++i)
  1201. {
  1202. // Balance
  1203. if (doBalance)
  1204. {
  1205. if (i % 2 == 0)
  1206. FLOAT_COPY(oldBufLeft, outBuffer[i]+timeOffset, frames);
  1207. float balRangeL = (pData->postProc.balanceLeft + 1.0f)/2.0f;
  1208. float balRangeR = (pData->postProc.balanceRight + 1.0f)/2.0f;
  1209. for (uint32_t k=0; k < frames; ++k)
  1210. {
  1211. if (i % 2 == 0)
  1212. {
  1213. // left
  1214. outBuffer[i][k+timeOffset] = oldBufLeft[k] * (1.0f - balRangeL);
  1215. outBuffer[i][k+timeOffset] += outBuffer[i+1][k+timeOffset] * (1.0f - balRangeR);
  1216. }
  1217. else
  1218. {
  1219. // right
  1220. outBuffer[i][k+timeOffset] = outBuffer[i][k+timeOffset] * balRangeR;
  1221. outBuffer[i][k+timeOffset] += oldBufLeft[k] * balRangeL;
  1222. }
  1223. }
  1224. }
  1225. // Volume
  1226. if (fUses16Outs)
  1227. {
  1228. for (uint32_t k=0; k < frames; ++k)
  1229. outBuffer[i][k+timeOffset] = fAudio16Buffers[i][k] * pData->postProc.volume;
  1230. }
  1231. else if (doVolume)
  1232. {
  1233. for (uint32_t k=0; k < frames; ++k)
  1234. outBuffer[i][k+timeOffset] *= pData->postProc.volume;
  1235. }
  1236. }
  1237. } // End of Post-processing
  1238. #else
  1239. if (fUses16Outs)
  1240. {
  1241. for (uint32_t i=0; i < pData->audioOut.count; ++i)
  1242. {
  1243. for (uint32_t k=0; k < frames; ++k)
  1244. outBuffer[i][k+timeOffset] = fAudio16Buffers[i][k];
  1245. }
  1246. }
  1247. #endif
  1248. // --------------------------------------------------------------------------------------------------------
  1249. pData->singleMutex.unlock();
  1250. return true;
  1251. }
  1252. void bufferSizeChanged(const uint32_t newBufferSize) override
  1253. {
  1254. if (! fUses16Outs)
  1255. return;
  1256. for (uint32_t i=0; i < pData->audioOut.count; ++i)
  1257. {
  1258. if (fAudio16Buffers[i] != nullptr)
  1259. delete[] fAudio16Buffers[i];
  1260. fAudio16Buffers[i] = new float[newBufferSize];
  1261. }
  1262. }
  1263. void sampleRateChanged(const double newSampleRate) override
  1264. {
  1265. CARLA_SAFE_ASSERT_RETURN(fSettings != nullptr,);
  1266. fluid_settings_setnum(fSettings, "synth.sample-rate", newSampleRate);
  1267. #ifdef FLUIDSYNTH_VERSION_NEW_API
  1268. CARLA_SAFE_ASSERT_RETURN(fSynth != nullptr,);
  1269. fluid_synth_set_sample_rate(fSynth, float(newSampleRate));
  1270. #endif
  1271. }
  1272. // -------------------------------------------------------------------
  1273. // Plugin buffers
  1274. void clearBuffers() override
  1275. {
  1276. carla_debug("FluidSynthPlugin::clearBuffers() - start");
  1277. if (fAudio16Buffers != nullptr)
  1278. {
  1279. for (uint32_t i=0; i < pData->audioOut.count; ++i)
  1280. {
  1281. if (fAudio16Buffers[i] != nullptr)
  1282. {
  1283. delete[] fAudio16Buffers[i];
  1284. fAudio16Buffers[i] = nullptr;
  1285. }
  1286. }
  1287. delete[] fAudio16Buffers;
  1288. fAudio16Buffers = nullptr;
  1289. }
  1290. CarlaPlugin::clearBuffers();
  1291. carla_debug("FluidSynthPlugin::clearBuffers() - end");
  1292. }
  1293. // -------------------------------------------------------------------
  1294. const void* getExtraStuff() const noexcept override
  1295. {
  1296. static const char xtrue[] = "true";
  1297. static const char xfalse[] = "false";
  1298. return fUses16Outs ? xtrue : xfalse;
  1299. }
  1300. bool init(const char* const filename, const char* const name, const char* const label)
  1301. {
  1302. CARLA_SAFE_ASSERT_RETURN(pData->engine != nullptr, false);
  1303. // ---------------------------------------------------------------
  1304. // first checks
  1305. if (pData->client != nullptr)
  1306. {
  1307. pData->engine->setLastError("Plugin client is already registered");
  1308. return false;
  1309. }
  1310. if (fSynth == nullptr)
  1311. {
  1312. pData->engine->setLastError("null synth");
  1313. return false;
  1314. }
  1315. if (filename == nullptr || filename[0] == '\0')
  1316. {
  1317. pData->engine->setLastError("null filename");
  1318. return false;
  1319. }
  1320. if (label == nullptr || label[0] == '\0')
  1321. {
  1322. pData->engine->setLastError("null label");
  1323. return false;
  1324. }
  1325. // ---------------------------------------------------------------
  1326. // open soundfont
  1327. const int synthId(fluid_synth_sfload(fSynth, filename, 0));
  1328. if (synthId < 0)
  1329. {
  1330. pData->engine->setLastError("Failed to load SoundFont file");
  1331. return false;
  1332. }
  1333. fSynthId = static_cast<uint>(synthId);
  1334. // ---------------------------------------------------------------
  1335. // get info
  1336. CarlaString label2(label);
  1337. if (fUses16Outs && ! label2.endsWith(" (16 outs)"))
  1338. label2 += " (16 outs)";
  1339. fLabel = label2.dup();
  1340. pData->filename = carla_strdup(filename);
  1341. if (name != nullptr && name[0] != '\0')
  1342. pData->name = pData->engine->getUniquePluginName(name);
  1343. else
  1344. pData->name = pData->engine->getUniquePluginName(label);
  1345. // ---------------------------------------------------------------
  1346. // register client
  1347. pData->client = pData->engine->addClient(this);
  1348. if (pData->client == nullptr || ! pData->client->isOk())
  1349. {
  1350. pData->engine->setLastError("Failed to register plugin client");
  1351. return false;
  1352. }
  1353. // ---------------------------------------------------------------
  1354. // load plugin settings
  1355. {
  1356. // set default options
  1357. pData->options = 0x0;
  1358. pData->options |= PLUGIN_OPTION_MAP_PROGRAM_CHANGES;
  1359. pData->options |= PLUGIN_OPTION_SEND_CHANNEL_PRESSURE;
  1360. pData->options |= PLUGIN_OPTION_SEND_PITCHBEND;
  1361. pData->options |= PLUGIN_OPTION_SEND_ALL_SOUND_OFF;
  1362. // set identifier string
  1363. CarlaString identifier("SF2/");
  1364. if (const char* const shortname = std::strrchr(filename, OS_SEP))
  1365. identifier += shortname+1;
  1366. else
  1367. identifier += label;
  1368. pData->identifier = identifier.dup();
  1369. // load settings
  1370. pData->options = pData->loadSettings(pData->options, getOptionsAvailable());
  1371. }
  1372. return true;
  1373. }
  1374. private:
  1375. enum FluidSynthInputParameters {
  1376. FluidSynthReverbOnOff = 0,
  1377. FluidSynthReverbRoomSize = 1,
  1378. FluidSynthReverbDamp = 2,
  1379. FluidSynthReverbLevel = 3,
  1380. FluidSynthReverbWidth = 4,
  1381. FluidSynthChorusOnOff = 5,
  1382. FluidSynthChorusNr = 6,
  1383. FluidSynthChorusLevel = 7,
  1384. FluidSynthChorusSpeedHz = 8,
  1385. FluidSynthChorusDepthMs = 9,
  1386. FluidSynthChorusType = 10,
  1387. FluidSynthPolyphony = 11,
  1388. FluidSynthInterpolation = 12,
  1389. FluidSynthVoiceCount = 13,
  1390. FluidSynthParametersMax = 14
  1391. };
  1392. const bool fUses16Outs;
  1393. fluid_settings_t* fSettings;
  1394. fluid_synth_t* fSynth;
  1395. uint fSynthId;
  1396. float** fAudio16Buffers;
  1397. float fParamBuffers[FluidSynthParametersMax];
  1398. int32_t fCurMidiProgs[MAX_MIDI_CHANNELS];
  1399. const char* fLabel;
  1400. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(FluidSynthPlugin)
  1401. };
  1402. CARLA_BACKEND_END_NAMESPACE
  1403. #endif // WANT_FLUIDSYNTH
  1404. CARLA_BACKEND_START_NAMESPACE
  1405. CarlaPlugin* CarlaPlugin::newFluidSynth(const Initializer& init, const bool use16Outs)
  1406. {
  1407. carla_debug("CarlaPlugin::newFluidSynth({%p, \"%s\", \"%s\", \"%s\"}, %s)", init.engine, init.filename, init.name, init.label, bool2str(use16Outs));
  1408. #ifdef WANT_FLUIDSYNTH
  1409. if (init.engine->getProccessMode() == ENGINE_PROCESS_MODE_CONTINUOUS_RACK && use16Outs)
  1410. {
  1411. init.engine->setLastError("Carla's rack mode can only work with Stereo modules, please choose the 2-channel only SoundFont version");
  1412. return nullptr;
  1413. }
  1414. if (! fluid_is_soundfont(init.filename))
  1415. {
  1416. init.engine->setLastError("Requested file is not a valid SoundFont");
  1417. return nullptr;
  1418. }
  1419. FluidSynthPlugin* const plugin(new FluidSynthPlugin(init.engine, init.id, use16Outs));
  1420. if (! plugin->init(init.filename, init.name, init.label))
  1421. {
  1422. delete plugin;
  1423. return nullptr;
  1424. }
  1425. plugin->reload();
  1426. return plugin;
  1427. #else
  1428. init.engine->setLastError("fluidsynth support not available");
  1429. return nullptr;
  1430. // unused
  1431. (void)use16Outs;
  1432. #endif
  1433. }
  1434. CARLA_BACKEND_END_NAMESPACE