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.

389 lines
13KB

  1. /*
  2. ZynAddSubFX - a software synthesizer
  3. Config.cpp - Configuration file functions
  4. Copyright (C) 2003-2005 Nasca Octavian Paul
  5. Author: Nasca Octavian Paul
  6. This program is free software; you can redistribute it and/or modify
  7. it under the terms of version 2 of the GNU General Public License
  8. as published by the Free Software Foundation.
  9. This program is distributed in the hope that it will be useful,
  10. but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. GNU General Public License (version 2 or later) for more details.
  13. You should have received a copy of the GNU General Public License (version 2)
  14. along with this program; if not, write to the Free Software Foundation,
  15. Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
  16. */
  17. #include <cstdio>
  18. #include <cmath>
  19. #include <cstdlib>
  20. #include <cstring>
  21. #include <rtosc/ports.h>
  22. #include <rtosc/port-sugar.h>
  23. #include "Config.h"
  24. #include "XMLwrapper.h"
  25. #define rStdString(name, len, ...) \
  26. {STRINGIFY(name) "::s", rMap(length, len) DOC(__VA_ARGS__), NULL, rStringCb(name,len)}
  27. #define rStdStringCb(name, length) rBOIL_BEGIN \
  28. if(!strcmp("", args)) {\
  29. data.reply(loc, "s", obj->name); \
  30. } else { \
  31. strncpy(obj->name, rtosc_argument(msg, 0).s, length); \
  32. data.broadcast(loc, "s", obj->name);\
  33. rChangeCb \
  34. } rBOIL_END
  35. #if 1
  36. #define rObject Config
  37. static rtosc::Ports ports = {
  38. //rString(cfg.LinuxOSSWaveOutDev),
  39. //rString(cfg.LinuxOSSSeqInDev),
  40. rParamI(cfg.SampleRate, "samples of audio per second"),
  41. rParamI(cfg.SoundBufferSize, "Size of processed audio buffer"),
  42. rParamI(cfg.OscilSize, "Size Of Oscillator Wavetable"),
  43. rToggle(cfg.SwapStereo, "Swap Left And Right Channels"),
  44. rToggle(cfg.BankUIAutoClose, "Automatic Closing of BackUI After Patch Selection"),
  45. rParamI(cfg.GzipCompression, "Level of Gzip Compression For Save Files"),
  46. rParamI(cfg.Interpolation, "Level of Interpolation, Linear/Cubic"),
  47. {"cfg.presetsDirList", rProp(parameter) rDoc("list of preset search directories"), 0,
  48. [](const char *msg, rtosc::RtData &d)
  49. {
  50. Config &c = *(Config*)d.obj;
  51. if(rtosc_narguments(msg) != 0)
  52. return;
  53. char types[MAX_BANK_ROOT_DIRS+1];
  54. rtosc_arg_t args[MAX_BANK_ROOT_DIRS];
  55. size_t pos = 0;
  56. //zero out data
  57. memset(types, 0, sizeof(types));
  58. memset(args, 0, sizeof(args));
  59. for(int i=0; i<MAX_BANK_ROOT_DIRS; ++i) {
  60. if(!c.cfg.presetsDirList[i].empty()) {
  61. types[pos] = 's';
  62. args[pos].s = c.cfg.presetsDirList[i].c_str();
  63. pos++;
  64. }
  65. }
  66. char buffer[1024*5];
  67. rtosc_amessage(buffer, sizeof(buffer), d.loc, types, args);
  68. d.reply(buffer);
  69. }},
  70. {"cfg.bankRootDirList", rProp(parameter) rDoc("list of bank search directories"), 0,
  71. [](const char *msg, rtosc::RtData &d)
  72. {
  73. Config &c = *(Config*)d.obj;
  74. if(rtosc_narguments(msg) != 0)
  75. return;
  76. char types[MAX_BANK_ROOT_DIRS+1];
  77. rtosc_arg_t args[MAX_BANK_ROOT_DIRS];
  78. size_t pos = 0;
  79. //zero out data
  80. memset(types, 0, sizeof(types));
  81. memset(args, 0, sizeof(args));
  82. for(int i=0; i<MAX_BANK_ROOT_DIRS; ++i) {
  83. if(!c.cfg.bankRootDirList[i].empty()) {
  84. types[pos] = 's';
  85. args[pos].s = c.cfg.bankRootDirList[i].c_str();
  86. pos++;
  87. }
  88. }
  89. char buffer[1024*5];
  90. rtosc_amessage(buffer, sizeof(buffer), d.loc, types, args);
  91. d.reply(buffer);
  92. }},
  93. //rArrayS(cfg.bankRootDirList,MAX_BANK_ROOT_DIRS),
  94. //rString(cfg.currentBankDir),
  95. //rArrayS(cfg.presetsDirList,MAX_BANK_ROOT_DIRS),
  96. rToggle(cfg.CheckPADsynth, "Old Check For PADsynth functionality within a patch"),
  97. rToggle(cfg.IgnoreProgramChange, "Ignore MIDI Program Change Events"),
  98. rParamI(cfg.UserInterfaceMode, "Beginner/Advanced Mode Select"),
  99. rParamI(cfg.VirKeybLayout, "Keyboard Layout For Virtual Piano Keyboard"),
  100. //rParamS(cfg.LinuxALSAaudioDev),
  101. //rParamS(cfg.nameTag)
  102. };
  103. rtosc::Ports &Config::ports = ::ports;
  104. #endif
  105. Config::Config()
  106. {}
  107. void Config::init()
  108. {
  109. maxstringsize = MAX_STRING_SIZE; //for ui
  110. //defaults
  111. cfg.SampleRate = 44100;
  112. cfg.SoundBufferSize = 256;
  113. cfg.OscilSize = 1024;
  114. cfg.SwapStereo = 0;
  115. cfg.LinuxOSSWaveOutDev = new char[MAX_STRING_SIZE];
  116. snprintf(cfg.LinuxOSSWaveOutDev, MAX_STRING_SIZE, "/dev/dsp");
  117. cfg.LinuxOSSSeqInDev = new char[MAX_STRING_SIZE];
  118. snprintf(cfg.LinuxOSSSeqInDev, MAX_STRING_SIZE, "/dev/sequencer");
  119. cfg.WindowsWaveOutId = 0;
  120. cfg.WindowsMidiInId = 0;
  121. cfg.BankUIAutoClose = 0;
  122. cfg.GzipCompression = 3;
  123. cfg.Interpolation = 0;
  124. cfg.CheckPADsynth = 1;
  125. cfg.IgnoreProgramChange = 0;
  126. cfg.UserInterfaceMode = 0;
  127. cfg.VirKeybLayout = 1;
  128. winwavemax = 1;
  129. winmidimax = 1;
  130. //try to find out how many input midi devices are there
  131. winmididevices = new winmidionedevice[winmidimax];
  132. for(int i = 0; i < winmidimax; ++i) {
  133. winmididevices[i].name = new char[MAX_STRING_SIZE];
  134. for(int j = 0; j < MAX_STRING_SIZE; ++j)
  135. winmididevices[i].name[j] = '\0';
  136. }
  137. //get the midi input devices name
  138. cfg.currentBankDir = "./testbnk";
  139. char filename[MAX_STRING_SIZE];
  140. getConfigFileName(filename, MAX_STRING_SIZE);
  141. readConfig(filename);
  142. if(cfg.bankRootDirList[0].empty()) {
  143. //banks
  144. cfg.bankRootDirList[0] = "~/banks";
  145. cfg.bankRootDirList[1] = "./";
  146. cfg.bankRootDirList[2] = "/usr/share/zynaddsubfx/banks";
  147. cfg.bankRootDirList[3] = "/usr/local/share/zynaddsubfx/banks";
  148. #ifdef __APPLE__
  149. cfg.bankRootDirList[4] = "../Resources/banks";
  150. #else
  151. cfg.bankRootDirList[4] = "../banks";
  152. #endif
  153. cfg.bankRootDirList[5] = "banks";
  154. }
  155. if(cfg.presetsDirList[0].empty()) {
  156. //presets
  157. cfg.presetsDirList[0] = "./";
  158. #ifdef __APPLE__
  159. cfg.presetsDirList[1] = "../Resources/presets";
  160. #else
  161. cfg.presetsDirList[1] = "../presets";
  162. #endif
  163. cfg.presetsDirList[2] = "presets";
  164. cfg.presetsDirList[3] = "/usr/share/zynaddsubfx/presets";
  165. cfg.presetsDirList[4] = "/usr/local/share/zynaddsubfx/presets";
  166. }
  167. cfg.LinuxALSAaudioDev = "default";
  168. cfg.nameTag = "";
  169. }
  170. Config::~Config()
  171. {
  172. delete [] cfg.LinuxOSSWaveOutDev;
  173. delete [] cfg.LinuxOSSSeqInDev;
  174. for(int i = 0; i < winmidimax; ++i)
  175. delete [] winmididevices[i].name;
  176. delete [] winmididevices;
  177. }
  178. void Config::save()
  179. {
  180. char filename[MAX_STRING_SIZE];
  181. getConfigFileName(filename, MAX_STRING_SIZE);
  182. saveConfig(filename);
  183. }
  184. void Config::clearbankrootdirlist()
  185. {
  186. for(int i = 0; i < MAX_BANK_ROOT_DIRS; ++i)
  187. cfg.bankRootDirList[i].clear();
  188. }
  189. void Config::clearpresetsdirlist()
  190. {
  191. for(int i = 0; i < MAX_BANK_ROOT_DIRS; ++i)
  192. cfg.presetsDirList[i].clear();
  193. }
  194. void Config::readConfig(const char *filename)
  195. {
  196. XMLwrapper xmlcfg;
  197. if(xmlcfg.loadXMLfile(filename) < 0)
  198. return;
  199. if(xmlcfg.enterbranch("CONFIGURATION")) {
  200. cfg.SampleRate = xmlcfg.getpar("sample_rate",
  201. cfg.SampleRate,
  202. 4000,
  203. 1024000);
  204. cfg.SoundBufferSize = xmlcfg.getpar("sound_buffer_size",
  205. cfg.SoundBufferSize,
  206. 16,
  207. 8192);
  208. cfg.OscilSize = xmlcfg.getpar("oscil_size",
  209. cfg.OscilSize,
  210. MAX_AD_HARMONICS * 2,
  211. 131072);
  212. cfg.SwapStereo = xmlcfg.getpar("swap_stereo",
  213. cfg.SwapStereo,
  214. 0,
  215. 1);
  216. cfg.BankUIAutoClose = xmlcfg.getpar("bank_window_auto_close",
  217. cfg.BankUIAutoClose,
  218. 0,
  219. 1);
  220. cfg.GzipCompression = xmlcfg.getpar("gzip_compression",
  221. cfg.GzipCompression,
  222. 0,
  223. 9);
  224. cfg.currentBankDir = xmlcfg.getparstr("bank_current", "");
  225. cfg.Interpolation = xmlcfg.getpar("interpolation",
  226. cfg.Interpolation,
  227. 0,
  228. 1);
  229. cfg.CheckPADsynth = xmlcfg.getpar("check_pad_synth",
  230. cfg.CheckPADsynth,
  231. 0,
  232. 1);
  233. cfg.IgnoreProgramChange = xmlcfg.getpar("ignore_program_change",
  234. cfg.IgnoreProgramChange,
  235. 0,
  236. 1);
  237. cfg.UserInterfaceMode = xmlcfg.getpar("user_interface_mode",
  238. cfg.UserInterfaceMode,
  239. 0,
  240. 2);
  241. cfg.VirKeybLayout = xmlcfg.getpar("virtual_keyboard_layout",
  242. cfg.VirKeybLayout,
  243. 0,
  244. 10);
  245. //get bankroot dirs
  246. for(int i = 0; i < MAX_BANK_ROOT_DIRS; ++i)
  247. if(xmlcfg.enterbranch("BANKROOT", i)) {
  248. cfg.bankRootDirList[i] = xmlcfg.getparstr("bank_root", "");
  249. xmlcfg.exitbranch();
  250. }
  251. //get preset root dirs
  252. for(int i = 0; i < MAX_BANK_ROOT_DIRS; ++i)
  253. if(xmlcfg.enterbranch("PRESETSROOT", i)) {
  254. cfg.presetsDirList[i] = xmlcfg.getparstr("presets_root", "");
  255. xmlcfg.exitbranch();
  256. }
  257. //linux stuff
  258. xmlcfg.getparstr("linux_oss_wave_out_dev",
  259. cfg.LinuxOSSWaveOutDev,
  260. MAX_STRING_SIZE);
  261. xmlcfg.getparstr("linux_oss_seq_in_dev",
  262. cfg.LinuxOSSSeqInDev,
  263. MAX_STRING_SIZE);
  264. //windows stuff
  265. cfg.WindowsWaveOutId = xmlcfg.getpar("windows_wave_out_id",
  266. cfg.WindowsWaveOutId,
  267. 0,
  268. winwavemax);
  269. cfg.WindowsMidiInId = xmlcfg.getpar("windows_midi_in_id",
  270. cfg.WindowsMidiInId,
  271. 0,
  272. winmidimax);
  273. xmlcfg.exitbranch();
  274. }
  275. cfg.OscilSize = (int) powf(2, ceil(logf(cfg.OscilSize - 1.0f) / logf(2.0f)));
  276. }
  277. void Config::saveConfig(const char *filename)
  278. {
  279. XMLwrapper *xmlcfg = new XMLwrapper();
  280. xmlcfg->beginbranch("CONFIGURATION");
  281. xmlcfg->addpar("sample_rate", cfg.SampleRate);
  282. xmlcfg->addpar("sound_buffer_size", cfg.SoundBufferSize);
  283. xmlcfg->addpar("oscil_size", cfg.OscilSize);
  284. xmlcfg->addpar("swap_stereo", cfg.SwapStereo);
  285. xmlcfg->addpar("bank_window_auto_close", cfg.BankUIAutoClose);
  286. xmlcfg->addpar("gzip_compression", cfg.GzipCompression);
  287. xmlcfg->addpar("check_pad_synth", cfg.CheckPADsynth);
  288. xmlcfg->addpar("ignore_program_change", cfg.IgnoreProgramChange);
  289. xmlcfg->addparstr("bank_current", cfg.currentBankDir);
  290. xmlcfg->addpar("user_interface_mode", cfg.UserInterfaceMode);
  291. xmlcfg->addpar("virtual_keyboard_layout", cfg.VirKeybLayout);
  292. for(int i = 0; i < MAX_BANK_ROOT_DIRS; ++i)
  293. if(!cfg.bankRootDirList[i].empty()) {
  294. xmlcfg->beginbranch("BANKROOT", i);
  295. xmlcfg->addparstr("bank_root", cfg.bankRootDirList[i]);
  296. xmlcfg->endbranch();
  297. }
  298. for(int i = 0; i < MAX_BANK_ROOT_DIRS; ++i)
  299. if(!cfg.presetsDirList[i].empty()) {
  300. xmlcfg->beginbranch("PRESETSROOT", i);
  301. xmlcfg->addparstr("presets_root", cfg.presetsDirList[i]);
  302. xmlcfg->endbranch();
  303. }
  304. xmlcfg->addpar("interpolation", cfg.Interpolation);
  305. //linux stuff
  306. xmlcfg->addparstr("linux_oss_wave_out_dev", cfg.LinuxOSSWaveOutDev);
  307. xmlcfg->addparstr("linux_oss_seq_in_dev", cfg.LinuxOSSSeqInDev);
  308. //windows stuff
  309. xmlcfg->addpar("windows_wave_out_id", cfg.WindowsWaveOutId);
  310. xmlcfg->addpar("windows_midi_in_id", cfg.WindowsMidiInId);
  311. xmlcfg->endbranch();
  312. int tmp = cfg.GzipCompression;
  313. cfg.GzipCompression = 0;
  314. xmlcfg->saveXMLfile(filename);
  315. cfg.GzipCompression = tmp;
  316. delete (xmlcfg);
  317. }
  318. void Config::getConfigFileName(char *name, int namesize)
  319. {
  320. name[0] = 0;
  321. snprintf(name, namesize, "%s%s", getenv("HOME"), "/.zynaddsubfxXML.cfg");
  322. }