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.

570 lines
21KB

  1. /*
  2. * Carla State utils
  3. * Copyright (C) 2012-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 "CarlaStateUtils.hpp"
  18. #include "CarlaBackendUtils.hpp"
  19. #include "CarlaMathUtils.hpp"
  20. #include "CarlaMIDI.h"
  21. #include <QtCore/QString>
  22. #include <QtXml/QDomNode>
  23. CARLA_BACKEND_START_NAMESPACE
  24. // -----------------------------------------------------------------------
  25. // StateParameter
  26. StateParameter::StateParameter() noexcept
  27. : index(-1),
  28. name(nullptr),
  29. symbol(nullptr),
  30. value(0.0f),
  31. midiChannel(0),
  32. midiCC(-1) {}
  33. StateParameter::~StateParameter()
  34. {
  35. if (name != nullptr)
  36. {
  37. delete[] name;
  38. name = nullptr;
  39. }
  40. if (symbol != nullptr)
  41. {
  42. delete[] symbol;
  43. symbol = nullptr;
  44. }
  45. }
  46. // -----------------------------------------------------------------------
  47. // StateCustomData
  48. StateCustomData::StateCustomData() noexcept
  49. : type(nullptr),
  50. key(nullptr),
  51. value(nullptr) {}
  52. StateCustomData::~StateCustomData()
  53. {
  54. if (type != nullptr)
  55. {
  56. delete[] type;
  57. type = nullptr;
  58. }
  59. if (key != nullptr)
  60. {
  61. delete[] key;
  62. key = nullptr;
  63. }
  64. if (value != nullptr)
  65. {
  66. delete[] value;
  67. value = nullptr;
  68. }
  69. }
  70. // -----------------------------------------------------------------------
  71. // SaveState
  72. SaveState::SaveState() noexcept
  73. : type(nullptr),
  74. name(nullptr),
  75. label(nullptr),
  76. binary(nullptr),
  77. uniqueID(0),
  78. active(false),
  79. dryWet(1.0f),
  80. volume(1.0f),
  81. balanceLeft(-1.0f),
  82. balanceRight(1.0f),
  83. panning(0.0f),
  84. ctrlChannel(-1),
  85. currentProgramIndex(-1),
  86. currentProgramName(nullptr),
  87. currentMidiBank(-1),
  88. currentMidiProgram(-1),
  89. chunk(nullptr) {}
  90. SaveState::~SaveState()
  91. {
  92. reset();
  93. }
  94. void SaveState::reset()
  95. {
  96. if (type != nullptr)
  97. {
  98. delete[] type;
  99. type = nullptr;
  100. }
  101. if (name != nullptr)
  102. {
  103. delete[] name;
  104. name = nullptr;
  105. }
  106. if (label != nullptr)
  107. {
  108. delete[] label;
  109. label = nullptr;
  110. }
  111. if (binary != nullptr)
  112. {
  113. delete[] binary;
  114. binary = nullptr;
  115. }
  116. if (currentProgramName != nullptr)
  117. {
  118. delete[] currentProgramName;
  119. currentProgramName = nullptr;
  120. }
  121. if (chunk != nullptr)
  122. {
  123. delete[] chunk;
  124. chunk = nullptr;
  125. }
  126. uniqueID = 0;
  127. active = false;
  128. dryWet = 1.0f;
  129. volume = 1.0f;
  130. balanceLeft = -1.0f;
  131. balanceRight = 1.0f;
  132. panning = 0.0f;
  133. ctrlChannel = -1;
  134. currentProgramIndex = -1;
  135. currentMidiBank = -1;
  136. currentMidiProgram = -1;
  137. for (StateParameterItenerator it = parameters.begin(); it.valid(); it.next())
  138. {
  139. StateParameter* const stateParameter(it.getValue());
  140. delete stateParameter;
  141. }
  142. for (StateCustomDataItenerator it = customData.begin(); it.valid(); it.next())
  143. {
  144. StateCustomData* const stateCustomData(it.getValue());
  145. delete stateCustomData;
  146. }
  147. parameters.clear();
  148. customData.clear();
  149. }
  150. // -----------------------------------------------------------------------
  151. // xmlSafeString
  152. static QString xmlSafeString(const QString& string, const bool toXml)
  153. {
  154. QString newString(string);
  155. if (toXml)
  156. return newString.replace("&","&amp;").replace("<","&lt;").replace(">","&gt;").replace("'","&apos;").replace("\"","&quot;");
  157. else
  158. return newString.replace("&amp;","&").replace("&lt;","<").replace("&gt;",">").replace("&apos;","'").replace("&quot;","\"");
  159. }
  160. static const char* xmlSafeStringCharDup(const QString& string, const bool toXml)
  161. {
  162. return carla_strdup(xmlSafeString(string, toXml).toUtf8().constData());
  163. }
  164. // -----------------------------------------------------------------------
  165. // fillSaveStateFromXmlNode
  166. void fillSaveStateFromXmlNode(SaveState& saveState, const QDomNode& xmlNode)
  167. {
  168. if (xmlNode.isNull())
  169. return;
  170. for (QDomNode node = xmlNode.firstChild(); ! node.isNull(); node = node.nextSibling())
  171. {
  172. QString tagName(node.toElement().tagName());
  173. // ---------------------------------------------------------------
  174. // Info
  175. if (tagName.compare("info", Qt::CaseInsensitive) == 0)
  176. {
  177. for (QDomNode xmlInfo = node.toElement().firstChild(); ! xmlInfo.isNull(); xmlInfo = xmlInfo.nextSibling())
  178. {
  179. const QString tag(xmlInfo.toElement().tagName());
  180. const QString text(xmlInfo.toElement().text().trimmed());
  181. if (tag.compare("type", Qt::CaseInsensitive) == 0)
  182. {
  183. saveState.type = xmlSafeStringCharDup(text, false);
  184. }
  185. else if (tag.compare("name", Qt::CaseInsensitive) == 0)
  186. {
  187. saveState.name = xmlSafeStringCharDup(text, false);
  188. }
  189. else if (tag.compare("label", Qt::CaseInsensitive) == 0 || tag.compare("uri", Qt::CaseInsensitive) == 0)
  190. {
  191. saveState.label = xmlSafeStringCharDup(text, false);
  192. }
  193. else if (tag.compare("binary", Qt::CaseInsensitive) == 0 || tag.compare("bundle", Qt::CaseInsensitive) == 0 || tag.compare("filename", Qt::CaseInsensitive) == 0)
  194. {
  195. saveState.binary = xmlSafeStringCharDup(text, false);
  196. }
  197. else if (tag.compare("uniqueid", Qt::CaseInsensitive) == 0)
  198. {
  199. bool ok;
  200. const long uniqueID(text.toLong(&ok));
  201. if (ok) saveState.uniqueID = uniqueID;
  202. }
  203. }
  204. }
  205. // ---------------------------------------------------------------
  206. // Data
  207. else if (tagName.compare("data", Qt::CaseInsensitive) == 0)
  208. {
  209. for (QDomNode xmlData = node.toElement().firstChild(); ! xmlData.isNull(); xmlData = xmlData.nextSibling())
  210. {
  211. const QString tag(xmlData.toElement().tagName());
  212. const QString text(xmlData.toElement().text().trimmed());
  213. // -------------------------------------------------------
  214. // Internal Data
  215. if (tag.compare("active", Qt::CaseInsensitive) == 0)
  216. {
  217. saveState.active = (text.compare("yes", Qt::CaseInsensitive) == 0 || text.compare("true", Qt::CaseInsensitive) == 0);
  218. }
  219. else if (tag.compare("drywet", Qt::CaseInsensitive) == 0)
  220. {
  221. bool ok;
  222. const float value(text.toFloat(&ok));
  223. if (ok) saveState.dryWet = carla_fixValue(0.0f, 1.0f, value);
  224. }
  225. else if (tag.compare("volume", Qt::CaseInsensitive) == 0)
  226. {
  227. bool ok;
  228. const float value(text.toFloat(&ok));
  229. if (ok) saveState.volume = carla_fixValue(0.0f, 1.27f, value);
  230. }
  231. else if (tag.compare("balanceleft", Qt::CaseInsensitive) == 0 || tag.compare("balance-left", Qt::CaseInsensitive) == 0)
  232. {
  233. bool ok;
  234. const float value(text.toFloat(&ok));
  235. if (ok) saveState.balanceLeft = carla_fixValue(-1.0f, 1.0f, value);
  236. }
  237. else if (tag.compare("balanceright", Qt::CaseInsensitive) == 0 || tag.compare("balance-right", Qt::CaseInsensitive) == 0)
  238. {
  239. bool ok;
  240. const float value(text.toFloat(&ok));
  241. if (ok) saveState.balanceRight = carla_fixValue(-1.0f, 1.0f, value);
  242. }
  243. else if (tag.compare("panning", Qt::CaseInsensitive) == 0)
  244. {
  245. bool ok;
  246. const float value(text.toFloat(&ok));
  247. if (ok) saveState.panning = carla_fixValue(-1.0f, 1.0f, value);
  248. }
  249. else if (tag.compare("controlchannel", Qt::CaseInsensitive) == 0 || tag.compare("control-channel", Qt::CaseInsensitive) == 0)
  250. {
  251. bool ok;
  252. const short value(text.toShort(&ok));
  253. if (ok && value >= 1 && value < MAX_MIDI_CHANNELS)
  254. saveState.ctrlChannel = static_cast<int8_t>(value-1);
  255. }
  256. // -------------------------------------------------------
  257. // Program (current)
  258. else if (tag.compare("currentprogramindex", Qt::CaseInsensitive) == 0 || tag.compare("current-program-index", Qt::CaseInsensitive) == 0)
  259. {
  260. bool ok;
  261. const int value(text.toInt(&ok));
  262. if (ok && value >= 1)
  263. saveState.currentProgramIndex = value-1;
  264. }
  265. else if (tag.compare("currentprogramname", Qt::CaseInsensitive) == 0 || tag.compare("current-program-name", Qt::CaseInsensitive) == 0)
  266. {
  267. saveState.currentProgramName = xmlSafeStringCharDup(text, false);
  268. }
  269. // -------------------------------------------------------
  270. // Midi Program (current)
  271. else if (tag.compare("currentmidibank", Qt::CaseInsensitive) == 0 || tag.compare("current-midi-bank", Qt::CaseInsensitive) == 0)
  272. {
  273. bool ok;
  274. const int value(text.toInt(&ok));
  275. if (ok && value >= 1)
  276. saveState.currentMidiBank = value-1;
  277. }
  278. else if (tag.compare("currentmidiprogram", Qt::CaseInsensitive) == 0 || tag.compare("current-midi-program", Qt::CaseInsensitive) == 0)
  279. {
  280. bool ok;
  281. const int value(text.toInt(&ok));
  282. if (ok && value >= 1)
  283. saveState.currentMidiProgram = value-1;
  284. }
  285. // -------------------------------------------------------
  286. // Parameters
  287. else if (tag.compare("parameter", Qt::CaseInsensitive) == 0)
  288. {
  289. StateParameter* const stateParameter(new StateParameter());
  290. for (QDomNode xmlSubData = xmlData.toElement().firstChild(); ! xmlSubData.isNull(); xmlSubData = xmlSubData.nextSibling())
  291. {
  292. const QString pTag(xmlSubData.toElement().tagName());
  293. const QString pText(xmlSubData.toElement().text().trimmed());
  294. if (pTag.compare("index", Qt::CaseInsensitive) == 0)
  295. {
  296. bool ok;
  297. const int index(pText.toInt(&ok));
  298. if (ok && index >= 0) stateParameter->index = index;
  299. }
  300. else if (pTag.compare("name", Qt::CaseInsensitive) == 0)
  301. {
  302. stateParameter->name = xmlSafeStringCharDup(pText, false);
  303. }
  304. else if (pTag.compare("symbol", Qt::CaseInsensitive) == 0)
  305. {
  306. stateParameter->symbol = xmlSafeStringCharDup(pText, false);
  307. }
  308. else if (pTag.compare("value", Qt::CaseInsensitive) == 0)
  309. {
  310. bool ok;
  311. const float value(pText.toFloat(&ok));
  312. if (ok) stateParameter->value = value;
  313. }
  314. else if (pTag.compare("midichannel", Qt::CaseInsensitive) == 0 || pTag.compare("midi-channel", Qt::CaseInsensitive) == 0)
  315. {
  316. bool ok;
  317. const ushort channel(pText.toUShort(&ok));
  318. if (ok && channel >= 1 && channel < MAX_MIDI_CHANNELS)
  319. stateParameter->midiChannel = static_cast<uint8_t>(channel-1);
  320. }
  321. else if (pTag.compare("midicc", Qt::CaseInsensitive) == 0 || pTag.compare("midi-cc", Qt::CaseInsensitive) == 0)
  322. {
  323. bool ok;
  324. const int cc(pText.toInt(&ok));
  325. if (ok && cc >= 1 && cc < 0x5F)
  326. stateParameter->midiCC = static_cast<int16_t>(cc);
  327. }
  328. }
  329. saveState.parameters.append(stateParameter);
  330. }
  331. // -------------------------------------------------------
  332. // Custom Data
  333. else if (tag.compare("customdata", Qt::CaseInsensitive) == 0 || tag.compare("custom-data", Qt::CaseInsensitive) == 0)
  334. {
  335. StateCustomData* const stateCustomData(new StateCustomData());
  336. for (QDomNode xmlSubData = xmlData.toElement().firstChild(); ! xmlSubData.isNull(); xmlSubData = xmlSubData.nextSibling())
  337. {
  338. const QString cTag(xmlSubData.toElement().tagName());
  339. const QString cText(xmlSubData.toElement().text().trimmed());
  340. if (cTag.compare("type", Qt::CaseInsensitive) == 0)
  341. stateCustomData->type = xmlSafeStringCharDup(cText, false);
  342. else if (cTag.compare("key", Qt::CaseInsensitive) == 0)
  343. stateCustomData->key = xmlSafeStringCharDup(cText, false);
  344. else if (cTag.compare("value", Qt::CaseInsensitive) == 0)
  345. stateCustomData->value = xmlSafeStringCharDup(cText, false);
  346. }
  347. saveState.customData.append(stateCustomData);
  348. }
  349. // -------------------------------------------------------
  350. // Chunk
  351. else if (tag.compare("chunk", Qt::CaseInsensitive) == 0)
  352. {
  353. saveState.chunk = xmlSafeStringCharDup(text, false);
  354. }
  355. }
  356. }
  357. }
  358. }
  359. // -----------------------------------------------------------------------
  360. // fillXmlStringFromSaveState
  361. void fillXmlStringFromSaveState(QString& content, const SaveState& saveState)
  362. {
  363. {
  364. QString info(" <Info>\n");
  365. info += QString(" <Type>%1</Type>\n").arg(saveState.type);
  366. info += QString(" <Name>%1</Name>\n").arg(xmlSafeString(saveState.name, true));
  367. switch (getPluginTypeFromString(saveState.type))
  368. {
  369. case PLUGIN_NONE:
  370. break;
  371. case PLUGIN_INTERNAL:
  372. info += QString(" <Label>%1</Label>\n").arg(xmlSafeString(saveState.label, true));
  373. break;
  374. case PLUGIN_LADSPA:
  375. info += QString(" <Binary>%1</Binary>\n").arg(xmlSafeString(saveState.binary, true));
  376. info += QString(" <Label>%1</Label>\n").arg(xmlSafeString(saveState.label, true));
  377. info += QString(" <UniqueID>%1</UniqueID>\n").arg(saveState.uniqueID);
  378. break;
  379. case PLUGIN_DSSI:
  380. info += QString(" <Binary>%1</Binary>\n").arg(xmlSafeString(saveState.binary, true));
  381. info += QString(" <Label>%1</Label>\n").arg(xmlSafeString(saveState.label, true));
  382. break;
  383. case PLUGIN_LV2:
  384. info += QString(" <Bundle>%1</Bundle>\n").arg(xmlSafeString(saveState.binary, true));
  385. info += QString(" <URI>%1</URI>\n").arg(xmlSafeString(saveState.label, true));
  386. break;
  387. case PLUGIN_VST:
  388. info += QString(" <Binary>%1</Binary>\n").arg(xmlSafeString(saveState.binary, true));
  389. info += QString(" <UniqueID>%1</UniqueID>\n").arg(saveState.uniqueID);
  390. break;
  391. case PLUGIN_AU:
  392. // TODO?
  393. info += QString(" <Binary>%1</Binary>\n").arg(xmlSafeString(saveState.binary, true));
  394. info += QString(" <UniqueID>%1</UniqueID>\n").arg(saveState.uniqueID);
  395. break;
  396. case PLUGIN_FILE_CSD:
  397. case PLUGIN_FILE_GIG:
  398. case PLUGIN_FILE_SF2:
  399. case PLUGIN_FILE_SFZ:
  400. info += QString(" <Filename>%1</Filename>\n").arg(xmlSafeString(saveState.binary, true));
  401. info += QString(" <Label>%1</Label>\n").arg(xmlSafeString(saveState.label, true));
  402. break;
  403. }
  404. info += " </Info>\n\n";
  405. content += info;
  406. }
  407. {
  408. QString data(" <Data>\n");
  409. data += QString(" <Active>%1</Active>\n").arg(saveState.active ? "Yes" : "No");
  410. if (saveState.dryWet != 1.0f)
  411. data += QString(" <DryWet>%1</DryWet>\n").arg(saveState.dryWet);
  412. if (saveState.volume != 1.0f)
  413. data += QString(" <Volume>%1</Volume>\n").arg(saveState.volume);
  414. if (saveState.balanceLeft != -1.0f)
  415. data += QString(" <Balance-Left>%1</Balance-Left>\n").arg(saveState.balanceLeft);
  416. if (saveState.balanceRight != 1.0f)
  417. data += QString(" <Balance-Right>%1</Balance-Right>\n").arg(saveState.balanceRight);
  418. if (saveState.panning != 0.0f)
  419. data += QString(" <Panning>%1</Panning>\n").arg(saveState.panning);
  420. if (saveState.ctrlChannel < 0)
  421. data += QString(" <ControlChannel>N</ControlChannel>\n");
  422. else
  423. data += QString(" <ControlChannel>%1</ControlChannel>\n").arg(saveState.ctrlChannel+1);
  424. content += data;
  425. }
  426. for (StateParameterItenerator it = saveState.parameters.begin(); it.valid(); it.next())
  427. {
  428. StateParameter* const stateParameter(it.getValue());
  429. QString parameter("\n"" <Parameter>\n");
  430. parameter += QString(" <Index>%1</Index>\n").arg(stateParameter->index);
  431. parameter += QString(" <Name>%1</Name>\n").arg(xmlSafeString(stateParameter->name, true));
  432. if (stateParameter->symbol != nullptr && stateParameter->symbol[0] != '\0')
  433. parameter += QString(" <Symbol>%1</Symbol>\n").arg(xmlSafeString(stateParameter->symbol, true));
  434. parameter += QString(" <Value>%1</Value>\n").arg(stateParameter->value);
  435. if (stateParameter->midiCC > 0)
  436. {
  437. parameter += QString(" <MidiCC>%1</MidiCC>\n").arg(stateParameter->midiCC);
  438. parameter += QString(" <MidiChannel>%1</MidiChannel>\n").arg(stateParameter->midiChannel+1);
  439. }
  440. parameter += " </Parameter>\n";
  441. content += parameter;
  442. }
  443. if (saveState.currentProgramIndex >= 0 && saveState.currentProgramName != nullptr && saveState.currentProgramName[0] != '\0')
  444. {
  445. // ignore 'default' program
  446. if (saveState.currentProgramIndex > 0 || QString(saveState.currentProgramName).compare("default", Qt::CaseInsensitive) != 0)
  447. {
  448. QString program("\n");
  449. program += QString(" <CurrentProgramIndex>%1</CurrentProgramIndex>\n").arg(saveState.currentProgramIndex+1);
  450. program += QString(" <CurrentProgramName>%1</CurrentProgramName>\n").arg(xmlSafeString(saveState.currentProgramName, true));
  451. content += program;
  452. }
  453. }
  454. if (saveState.currentMidiBank >= 0 && saveState.currentMidiProgram >= 0)
  455. {
  456. QString midiProgram("\n");
  457. midiProgram += QString(" <CurrentMidiBank>%1</CurrentMidiBank>\n").arg(saveState.currentMidiBank+1);
  458. midiProgram += QString(" <CurrentMidiProgram>%1</CurrentMidiProgram>\n").arg(saveState.currentMidiProgram+1);
  459. content += midiProgram;
  460. }
  461. for (StateCustomDataItenerator it = saveState.customData.begin(); it.valid(); it.next())
  462. {
  463. StateCustomData* const stateCustomData(it.getValue());
  464. QString customData("\n"" <CustomData>\n");
  465. customData += QString(" <Type>%1</Type>\n").arg(xmlSafeString(stateCustomData->type, true));
  466. customData += QString(" <Key>%1</Key>\n").arg(xmlSafeString(stateCustomData->key, true));
  467. if (std::strcmp(stateCustomData->type, CUSTOM_DATA_TYPE_CHUNK) == 0 || std::strlen(stateCustomData->value) >= 128)
  468. {
  469. customData += " <Value>\n";
  470. customData += QString("%1\n").arg(xmlSafeString(stateCustomData->value, true));
  471. customData += " </Value>\n";
  472. }
  473. else
  474. customData += QString(" <Value>%1</Value>\n").arg(xmlSafeString(stateCustomData->value, true));
  475. customData += " </CustomData>\n";
  476. content += customData;
  477. }
  478. if (saveState.chunk != nullptr && saveState.chunk[0] != '\0')
  479. {
  480. QString chunk("\n"" <Chunk>\n");
  481. chunk += QString("%1\n").arg(saveState.chunk);
  482. chunk += " </Chunk>\n";
  483. content += chunk;
  484. }
  485. content += " </Data>\n";
  486. }
  487. // -----------------------------------------------------------------------
  488. CARLA_BACKEND_END_NAMESPACE