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.

763 lines
27KB

  1. // SPDX-FileCopyrightText: 2011-2025 Filipe Coelho <falktx@falktx.com>
  2. // SPDX-License-Identifier: GPL-2.0-or-later
  3. #include "CarlaStateUtils.hpp"
  4. #include "CarlaBackendUtils.hpp"
  5. #include "CarlaMathUtils.hpp"
  6. #include "CarlaMIDI.h"
  7. #include "water/streams/MemoryOutputStream.h"
  8. #include "water/xml/XmlElement.h"
  9. #include <string>
  10. using water::MemoryOutputStream;
  11. using water::XmlElement;
  12. CARLA_BACKEND_START_NAMESPACE
  13. // -----------------------------------------------------------------------
  14. // getNewLineSplittedString
  15. static void getNewLineSplittedString(MemoryOutputStream& stream, const water::String& string)
  16. {
  17. static const int kLineWidth = 120;
  18. int i = 0;
  19. const int length = string.length();
  20. const char* const raw = string.toUTF8();
  21. stream.preallocate(static_cast<std::size_t>(length + length/kLineWidth + 3));
  22. for (; i+kLineWidth < length; i += kLineWidth)
  23. {
  24. stream.write(raw+i, kLineWidth);
  25. stream.writeByte('\n');
  26. }
  27. stream << (raw+i);
  28. }
  29. // -----------------------------------------------------------------------
  30. // xmlSafeStringFast
  31. /* Based on some code by James Kanze from stackoverflow
  32. * https://stackoverflow.com/questions/7724011/in-c-whats-the-fastest-way-to-replace-all-occurrences-of-a-substring-within */
  33. static std::string replaceStdString(const std::string& original, const std::string& before, const std::string& after)
  34. {
  35. std::string::const_iterator current = original.begin(), end = original.end(), next;
  36. std::string retval;
  37. for (; (next = std::search(current, end, before.begin(), before.end())) != end;)
  38. {
  39. retval.append(current, next);
  40. retval.append(after);
  41. current = next + static_cast<ssize_t>(before.size());
  42. }
  43. retval.append(current, next);
  44. return retval;
  45. }
  46. static std::string xmlSafeStringFast(const char* const cstring, const bool toXml)
  47. {
  48. std::string string(cstring);
  49. if (toXml)
  50. {
  51. string = replaceStdString(string, "&","&amp;");
  52. string = replaceStdString(string, "<","&lt;");
  53. string = replaceStdString(string, ">","&gt;");
  54. string = replaceStdString(string, "'","&apos;");
  55. string = replaceStdString(string, "\"","&quot;");
  56. }
  57. else
  58. {
  59. string = replaceStdString(string, "&lt;","<");
  60. string = replaceStdString(string, "&gt;",">");
  61. string = replaceStdString(string, "&apos;","'");
  62. string = replaceStdString(string, "&quot;","\"");
  63. string = replaceStdString(string, "&amp;","&");
  64. }
  65. return string;
  66. }
  67. // -----------------------------------------------------------------------
  68. // xmlSafeStringCharDup
  69. /*
  70. static const char* xmlSafeStringCharDup(const char* const cstring, const bool toXml)
  71. {
  72. return carla_strdup(xmlSafeString(cstring, toXml).toRawUTF8());
  73. }
  74. */
  75. static const char* xmlSafeStringCharDup(const water::String& string, const bool toXml)
  76. {
  77. return carla_strdup(xmlSafeString(string, toXml).toRawUTF8());
  78. }
  79. // -----------------------------------------------------------------------
  80. // StateParameter
  81. CarlaStateSave::Parameter::Parameter() noexcept
  82. : dummy(true),
  83. index(-1),
  84. name(nullptr),
  85. symbol(nullptr),
  86. #ifndef BUILD_BRIDGE_ALTERNATIVE_ARCH
  87. value(0.0f),
  88. mappedControlIndex(CONTROL_INDEX_NONE),
  89. midiChannel(0),
  90. mappedRangeValid(false),
  91. mappedMinimum(0.0f),
  92. mappedMaximum(1.0f) {}
  93. #else
  94. value(0.0f) {}
  95. #endif
  96. CarlaStateSave::Parameter::~Parameter() noexcept
  97. {
  98. if (name != nullptr)
  99. {
  100. delete[] name;
  101. name = nullptr;
  102. }
  103. if (symbol != nullptr)
  104. {
  105. delete[] symbol;
  106. symbol = nullptr;
  107. }
  108. }
  109. // -----------------------------------------------------------------------
  110. // StateCustomData
  111. CarlaStateSave::CustomData::CustomData() noexcept
  112. : type(nullptr),
  113. key(nullptr),
  114. value(nullptr) {}
  115. CarlaStateSave::CustomData::~CustomData() noexcept
  116. {
  117. if (type != nullptr)
  118. {
  119. delete[] type;
  120. type = nullptr;
  121. }
  122. if (key != nullptr)
  123. {
  124. delete[] key;
  125. key = nullptr;
  126. }
  127. if (value != nullptr)
  128. {
  129. delete[] value;
  130. value = nullptr;
  131. }
  132. }
  133. bool CarlaStateSave::CustomData::isValid() const noexcept
  134. {
  135. if (type == nullptr || type[0] == '\0') return false;
  136. if (key == nullptr || key [0] == '\0') return false;
  137. if (value == nullptr) return false;
  138. return true;
  139. }
  140. // -----------------------------------------------------------------------
  141. // StateSave
  142. CarlaStateSave::CarlaStateSave() noexcept
  143. : type(nullptr),
  144. name(nullptr),
  145. label(nullptr),
  146. binary(nullptr),
  147. uniqueId(0),
  148. options(PLUGIN_OPTIONS_NULL),
  149. temporary(false),
  150. #ifndef BUILD_BRIDGE_ALTERNATIVE_ARCH
  151. active(false),
  152. dryWet(1.0f),
  153. volume(1.0f),
  154. balanceLeft(-1.0f),
  155. balanceRight(1.0f),
  156. panning(0.0f),
  157. ctrlChannel(-1),
  158. #endif
  159. currentProgramIndex(-1),
  160. currentProgramName(nullptr),
  161. currentMidiBank(-1),
  162. currentMidiProgram(-1),
  163. chunk(nullptr),
  164. parameters(),
  165. customData() {}
  166. CarlaStateSave::~CarlaStateSave() noexcept
  167. {
  168. clear();
  169. }
  170. void CarlaStateSave::clear() noexcept
  171. {
  172. if (type != nullptr)
  173. {
  174. delete[] type;
  175. type = nullptr;
  176. }
  177. if (name != nullptr)
  178. {
  179. delete[] name;
  180. name = nullptr;
  181. }
  182. if (label != nullptr)
  183. {
  184. delete[] label;
  185. label = nullptr;
  186. }
  187. if (binary != nullptr)
  188. {
  189. delete[] binary;
  190. binary = nullptr;
  191. }
  192. if (currentProgramName != nullptr)
  193. {
  194. delete[] currentProgramName;
  195. currentProgramName = nullptr;
  196. }
  197. if (chunk != nullptr)
  198. {
  199. delete[] chunk;
  200. chunk = nullptr;
  201. }
  202. uniqueId = 0;
  203. options = PLUGIN_OPTIONS_NULL;
  204. #ifndef BUILD_BRIDGE_ALTERNATIVE_ARCH
  205. active = false;
  206. dryWet = 1.0f;
  207. volume = 1.0f;
  208. balanceLeft = -1.0f;
  209. balanceRight = 1.0f;
  210. panning = 0.0f;
  211. ctrlChannel = -1;
  212. #endif
  213. currentProgramIndex = -1;
  214. currentMidiBank = -1;
  215. currentMidiProgram = -1;
  216. for (ParameterItenerator it = parameters.begin2(); it.valid(); it.next())
  217. {
  218. Parameter* const stateParameter(it.getValue(nullptr));
  219. delete stateParameter;
  220. }
  221. for (CustomDataItenerator it = customData.begin2(); it.valid(); it.next())
  222. {
  223. CustomData* const stateCustomData(it.getValue(nullptr));
  224. delete stateCustomData;
  225. }
  226. parameters.clear();
  227. customData.clear();
  228. }
  229. // -----------------------------------------------------------------------
  230. // fillFromXmlElement
  231. bool CarlaStateSave::fillFromXmlElement(const XmlElement* const xmlElement)
  232. {
  233. CARLA_SAFE_ASSERT_RETURN(xmlElement != nullptr, false);
  234. clear();
  235. for (XmlElement* elem = xmlElement->getFirstChildElement(); elem != nullptr; elem = elem->getNextElement())
  236. {
  237. const water::String& tagName(elem->getTagName());
  238. // ---------------------------------------------------------------
  239. // Info
  240. if (tagName == "Info")
  241. {
  242. for (XmlElement* xmlInfo = elem->getFirstChildElement(); xmlInfo != nullptr; xmlInfo = xmlInfo->getNextElement())
  243. {
  244. const water::String& tag(xmlInfo->getTagName());
  245. const water::String text(xmlInfo->getAllSubText().trim());
  246. /**/ if (tag == "Type")
  247. type = xmlSafeStringCharDup(text, false);
  248. else if (tag == "Name")
  249. name = xmlSafeStringCharDup(text, false);
  250. else if (tag == "Label" || tag == "URI" || tag == "Identifier" || tag == "Setup")
  251. label = xmlSafeStringCharDup(text, false);
  252. else if (tag == "Binary" || tag == "Bundle" || tag == "Filename")
  253. binary = xmlSafeStringCharDup(text, false);
  254. else if (tag == "UniqueID")
  255. uniqueId = text.getLargeIntValue();
  256. }
  257. }
  258. // ---------------------------------------------------------------
  259. // Data
  260. else if (tagName == "Data")
  261. {
  262. for (XmlElement* xmlData = elem->getFirstChildElement(); xmlData != nullptr; xmlData = xmlData->getNextElement())
  263. {
  264. const water::String& tag(xmlData->getTagName());
  265. const water::String text(xmlData->getAllSubText().trim());
  266. // -------------------------------------------------------
  267. // Internal Data
  268. /**/ if (tag == "Options")
  269. {
  270. const int value(text.getHexValue32());
  271. if (value > 0)
  272. options = static_cast<uint>(value);
  273. }
  274. #ifndef BUILD_BRIDGE_ALTERNATIVE_ARCH
  275. else if (tag == "Active")
  276. {
  277. active = (text == "Yes");
  278. }
  279. else if (tag == "DryWet")
  280. {
  281. dryWet = carla_fixedValue(0.0f, 1.0f, text.getFloatValue());
  282. }
  283. else if (tag == "Volume")
  284. {
  285. volume = carla_fixedValue(0.0f, 1.27f, text.getFloatValue());
  286. }
  287. else if (tag == "Balance-Left")
  288. {
  289. balanceLeft = carla_fixedValue(-1.0f, 1.0f, text.getFloatValue());
  290. }
  291. else if (tag == "Balance-Right")
  292. {
  293. balanceRight = carla_fixedValue(-1.0f, 1.0f, text.getFloatValue());
  294. }
  295. else if (tag == "Panning")
  296. {
  297. panning = carla_fixedValue(-1.0f, 1.0f, text.getFloatValue());
  298. }
  299. else if (tag == "ControlChannel")
  300. {
  301. if (! text.startsWithIgnoreCase("n"))
  302. {
  303. const int value(text.getIntValue());
  304. if (value >= 1 && value <= MAX_MIDI_CHANNELS)
  305. ctrlChannel = static_cast<int8_t>(value-1);
  306. }
  307. }
  308. #endif
  309. // -------------------------------------------------------
  310. // Program (current)
  311. else if (tag == "CurrentProgramIndex")
  312. {
  313. const int value(text.getIntValue());
  314. if (value >= 1)
  315. currentProgramIndex = value-1;
  316. }
  317. else if (tag == "CurrentProgramName")
  318. {
  319. currentProgramName = xmlSafeStringCharDup(text, false);
  320. }
  321. // -------------------------------------------------------
  322. // Midi Program (current)
  323. else if (tag == "CurrentMidiBank")
  324. {
  325. const int value(text.getIntValue());
  326. if (value >= 1)
  327. currentMidiBank = value-1;
  328. }
  329. else if (tag == "CurrentMidiProgram")
  330. {
  331. const int value(text.getIntValue());
  332. if (value >= 1)
  333. currentMidiProgram = value-1;
  334. }
  335. // -------------------------------------------------------
  336. // Parameters
  337. else if (tag == "Parameter")
  338. {
  339. Parameter* const stateParameter(new Parameter());
  340. #ifndef BUILD_BRIDGE_ALTERNATIVE_ARCH
  341. bool hasMappedMinimum = false, hasMappedMaximum = false;
  342. #endif
  343. for (XmlElement* xmlSubData = xmlData->getFirstChildElement(); xmlSubData != nullptr; xmlSubData = xmlSubData->getNextElement())
  344. {
  345. const water::String& pTag(xmlSubData->getTagName());
  346. const water::String pText(xmlSubData->getAllSubText().trim());
  347. /**/ if (pTag == "Index")
  348. {
  349. const int index(pText.getIntValue());
  350. if (index >= 0)
  351. stateParameter->index = index;
  352. }
  353. else if (pTag == "Name")
  354. {
  355. stateParameter->name = xmlSafeStringCharDup(pText, false);
  356. }
  357. else if (pTag == "Symbol" || pTag == "Identifier")
  358. {
  359. stateParameter->symbol = xmlSafeStringCharDup(pText, false);
  360. }
  361. else if (pTag == "Value")
  362. {
  363. stateParameter->dummy = false;
  364. stateParameter->value = pText.getFloatValue();
  365. }
  366. #ifndef BUILD_BRIDGE_ALTERNATIVE_ARCH
  367. else if (pTag == "MidiChannel")
  368. {
  369. const int channel(pText.getIntValue());
  370. if (channel >= 1 && channel <= MAX_MIDI_CHANNELS)
  371. stateParameter->midiChannel = static_cast<uint8_t>(channel-1);
  372. }
  373. else if (pTag == "MidiCC")
  374. {
  375. const int cc(pText.getIntValue());
  376. if (cc > 0 && cc < MAX_MIDI_CONTROL)
  377. stateParameter->mappedControlIndex = static_cast<int16_t>(cc);
  378. }
  379. else if (pTag == "MappedControlIndex")
  380. {
  381. const int ctrl(pText.getIntValue());
  382. if (ctrl > CONTROL_INDEX_NONE && ctrl <= CONTROL_INDEX_MAX_ALLOWED)
  383. if (ctrl != CONTROL_INDEX_MIDI_LEARN)
  384. stateParameter->mappedControlIndex = static_cast<int16_t>(ctrl);
  385. }
  386. else if (pTag == "MappedMinimum")
  387. {
  388. hasMappedMinimum = true;
  389. stateParameter->mappedMinimum = pText.getFloatValue();
  390. }
  391. else if (pTag == "MappedMaximum")
  392. {
  393. hasMappedMaximum = true;
  394. stateParameter->mappedMaximum = pText.getFloatValue();
  395. }
  396. #endif
  397. }
  398. #ifndef BUILD_BRIDGE_ALTERNATIVE_ARCH
  399. if (hasMappedMinimum && hasMappedMaximum)
  400. stateParameter->mappedRangeValid = true;
  401. #endif
  402. parameters.append(stateParameter);
  403. }
  404. // -------------------------------------------------------
  405. // Custom Data
  406. else if (tag == "CustomData")
  407. {
  408. CustomData* const stateCustomData(new CustomData());
  409. // find type first
  410. for (XmlElement* xmlSubData = xmlData->getFirstChildElement(); xmlSubData != nullptr; xmlSubData = xmlSubData->getNextElement())
  411. {
  412. const water::String& cTag(xmlSubData->getTagName());
  413. if (cTag != "Type")
  414. continue;
  415. stateCustomData->type = xmlSafeStringCharDup(xmlSubData->getAllSubText().trim(), false);
  416. break;
  417. }
  418. if (stateCustomData->type == nullptr || stateCustomData->type[0] == '\0')
  419. {
  420. carla_stderr("Reading CustomData type failed");
  421. delete stateCustomData;
  422. continue;
  423. }
  424. // now fill in key and value, knowing what the type is
  425. for (XmlElement* xmlSubData = xmlData->getFirstChildElement(); xmlSubData != nullptr; xmlSubData = xmlSubData->getNextElement())
  426. {
  427. const water::String& cTag(xmlSubData->getTagName());
  428. water::String cText(xmlSubData->getAllSubText());
  429. /**/ if (cTag == "Key")
  430. {
  431. stateCustomData->key = xmlSafeStringCharDup(cText.trim(), false);
  432. }
  433. else if (cTag == "Value")
  434. {
  435. // save operation adds a newline and newline+space around the string in some cases
  436. const int len = cText.length();
  437. if (std::strcmp(stateCustomData->type, CUSTOM_DATA_TYPE_CHUNK) == 0 || len >= 128+6)
  438. {
  439. CARLA_SAFE_ASSERT_CONTINUE(len >= 6);
  440. cText = cText.substring(1, len - 5);
  441. }
  442. stateCustomData->value = xmlSafeStringCharDup(cText, false);
  443. }
  444. }
  445. if (stateCustomData->isValid())
  446. {
  447. customData.append(stateCustomData);
  448. }
  449. else
  450. {
  451. carla_stderr("Reading CustomData property failed, missing data");
  452. delete stateCustomData;
  453. }
  454. }
  455. // -------------------------------------------------------
  456. // Chunk
  457. else if (tag == "Chunk")
  458. {
  459. chunk = carla_strdup(text.toRawUTF8());
  460. }
  461. }
  462. }
  463. }
  464. return true;
  465. }
  466. // -----------------------------------------------------------------------
  467. // fillXmlStringFromStateSave
  468. void CarlaStateSave::dumpToMemoryStream(MemoryOutputStream& content) const
  469. {
  470. const PluginType pluginType = getPluginTypeFromString(type);
  471. {
  472. MemoryOutputStream infoXml;
  473. infoXml << " <Info>\n";
  474. infoXml << " <Type>" << water::String(type != nullptr ? type : "") << "</Type>\n";
  475. infoXml << " <Name>" << xmlSafeString(name, true) << "</Name>\n";
  476. switch (pluginType)
  477. {
  478. case PLUGIN_NONE:
  479. break;
  480. case PLUGIN_INTERNAL:
  481. infoXml << " <Label>" << xmlSafeString(label, true) << "</Label>\n";
  482. break;
  483. case PLUGIN_LADSPA:
  484. infoXml << " <Binary>" << xmlSafeString(binary, true) << "</Binary>\n";
  485. infoXml << " <Label>" << xmlSafeString(label, true) << "</Label>\n";
  486. infoXml << " <UniqueID>" << water::int64(uniqueId) << "</UniqueID>\n";
  487. break;
  488. case PLUGIN_DSSI:
  489. infoXml << " <Binary>" << xmlSafeString(binary, true) << "</Binary>\n";
  490. infoXml << " <Label>" << xmlSafeString(label, true) << "</Label>\n";
  491. break;
  492. case PLUGIN_LV2:
  493. infoXml << " <URI>" << xmlSafeString(label, true) << "</URI>\n";
  494. break;
  495. case PLUGIN_VST2:
  496. infoXml << " <Binary>" << xmlSafeString(binary, true) << "</Binary>\n";
  497. infoXml << " <UniqueID>" << water::int64(uniqueId) << "</UniqueID>\n";
  498. break;
  499. case PLUGIN_VST3:
  500. infoXml << " <Binary>" << xmlSafeString(binary, true) << "</Binary>\n";
  501. infoXml << " <Label>" << xmlSafeString(label, true) << "</Label>\n";
  502. break;
  503. case PLUGIN_AU:
  504. infoXml << " <Bundle>" << xmlSafeString(binary, true) << "</Bundle>\n";
  505. infoXml << " <Identifier>" << xmlSafeString(label, true) << "</Identifier>\n";
  506. break;
  507. case PLUGIN_CLAP:
  508. infoXml << " <Binary>" << xmlSafeString(binary, true) << "</Binary>\n";
  509. infoXml << " <Identifier>" << xmlSafeString(label, true) << "</Identifier>\n";
  510. break;
  511. case PLUGIN_DLS:
  512. case PLUGIN_GIG:
  513. case PLUGIN_SF2:
  514. case PLUGIN_JSFX:
  515. infoXml << " <Filename>" << xmlSafeString(binary, true) << "</Filename>\n";
  516. infoXml << " <Label>" << xmlSafeString(label, true) << "</Label>\n";
  517. break;
  518. case PLUGIN_SFZ:
  519. infoXml << " <Filename>" << xmlSafeString(binary, true) << "</Filename>\n";
  520. break;
  521. case PLUGIN_JACK:
  522. infoXml << " <Filename>" << xmlSafeString(binary, true) << "</Filename>\n";
  523. infoXml << " <Setup>" << xmlSafeString(label, true) << "</Setup>\n";
  524. break;
  525. case PLUGIN_TYPE_COUNT:
  526. break;
  527. }
  528. infoXml << " </Info>\n\n";
  529. content << infoXml;
  530. }
  531. content << " <Data>\n";
  532. #ifndef BUILD_BRIDGE_ALTERNATIVE_ARCH
  533. {
  534. MemoryOutputStream dataXml;
  535. dataXml << " <Active>" << (active ? "Yes" : "No") << "</Active>\n";
  536. if (carla_isNotEqual(dryWet, 1.0f))
  537. dataXml << " <DryWet>" << water::String(dryWet, 7) << "</DryWet>\n";
  538. if (carla_isNotEqual(volume, 1.0f))
  539. dataXml << " <Volume>" << water::String(volume, 7) << "</Volume>\n";
  540. if (carla_isNotEqual(balanceLeft, -1.0f))
  541. dataXml << " <Balance-Left>" << water::String(balanceLeft, 7) << "</Balance-Left>\n";
  542. if (carla_isNotEqual(balanceRight, 1.0f))
  543. dataXml << " <Balance-Right>" << water::String(balanceRight, 7) << "</Balance-Right>\n";
  544. if (carla_isNotEqual(panning, 0.0f))
  545. dataXml << " <Panning>" << water::String(panning, 7) << "</Panning>\n";
  546. if (ctrlChannel < 0)
  547. dataXml << " <ControlChannel>N</ControlChannel>\n";
  548. else
  549. dataXml << " <ControlChannel>" << int(ctrlChannel+1) << "</ControlChannel>\n";
  550. dataXml << " <Options>0x" << water::String::toHexString(static_cast<int>(options)) << "</Options>\n";
  551. content << dataXml;
  552. }
  553. #endif
  554. for (ParameterItenerator it = parameters.begin2(); it.valid(); it.next())
  555. {
  556. Parameter* const stateParameter(it.getValue(nullptr));
  557. CARLA_SAFE_ASSERT_CONTINUE(stateParameter != nullptr);
  558. MemoryOutputStream parameterXml;
  559. parameterXml << "\n";
  560. parameterXml << " <Parameter>\n";
  561. parameterXml << " <Index>" << water::String(stateParameter->index) << "</Index>\n";
  562. parameterXml << " <Name>" << xmlSafeString(stateParameter->name, true) << "</Name>\n";
  563. if (stateParameter->symbol != nullptr && stateParameter->symbol[0] != '\0')
  564. {
  565. if (pluginType == PLUGIN_CLAP)
  566. parameterXml << " <Identifier>" << xmlSafeString(stateParameter->symbol, true) << "</Identifier>\n";
  567. else
  568. parameterXml << " <Symbol>" << xmlSafeString(stateParameter->symbol, true) << "</Symbol>\n";
  569. }
  570. #ifndef BUILD_BRIDGE_ALTERNATIVE_ARCH
  571. if (stateParameter->mappedControlIndex > CONTROL_INDEX_NONE && stateParameter->mappedControlIndex <= CONTROL_INDEX_MAX_ALLOWED)
  572. {
  573. parameterXml << " <MidiChannel>" << stateParameter->midiChannel+1 << "</MidiChannel>\n";
  574. parameterXml << " <MappedControlIndex>" << stateParameter->mappedControlIndex << "</MappedControlIndex>\n";
  575. if (stateParameter->mappedRangeValid)
  576. {
  577. parameterXml << " <MappedMinimum>" << water::String(stateParameter->mappedMinimum, 15) << "</MappedMinimum>\n";
  578. parameterXml << " <MappedMaximum>" << water::String(stateParameter->mappedMaximum, 15) << "</MappedMaximum>\n";
  579. }
  580. // backwards compatibility for older carla versions
  581. if (stateParameter->mappedControlIndex > 0 && stateParameter->mappedControlIndex < MAX_MIDI_CONTROL)
  582. parameterXml << " <MidiCC>" << stateParameter->mappedControlIndex << "</MidiCC>\n";
  583. }
  584. #endif
  585. if (! stateParameter->dummy)
  586. parameterXml << " <Value>" << water::String(stateParameter->value, 15) << "</Value>\n";
  587. parameterXml << " </Parameter>\n";
  588. content << parameterXml;
  589. }
  590. if (currentProgramIndex >= 0 && currentProgramName != nullptr && currentProgramName[0] != '\0')
  591. {
  592. // ignore 'default' program
  593. if (currentProgramIndex > 0 || ! water::String(currentProgramName).equalsIgnoreCase("default"))
  594. {
  595. MemoryOutputStream programXml;
  596. programXml << "\n";
  597. programXml << " <CurrentProgramIndex>" << currentProgramIndex+1 << "</CurrentProgramIndex>\n";
  598. programXml << " <CurrentProgramName>" << xmlSafeString(currentProgramName, true) << "</CurrentProgramName>\n";
  599. content << programXml;
  600. }
  601. }
  602. if (currentMidiBank >= 0 && currentMidiProgram >= 0)
  603. {
  604. MemoryOutputStream midiProgramXml;
  605. midiProgramXml << "\n";
  606. midiProgramXml << " <CurrentMidiBank>" << currentMidiBank+1 << "</CurrentMidiBank>\n";
  607. midiProgramXml << " <CurrentMidiProgram>" << currentMidiProgram+1 << "</CurrentMidiProgram>\n";
  608. content << midiProgramXml;
  609. }
  610. for (CustomDataItenerator it = customData.begin2(); it.valid(); it.next())
  611. {
  612. CustomData* const stateCustomData(it.getValue(nullptr));
  613. CARLA_SAFE_ASSERT_CONTINUE(stateCustomData != nullptr);
  614. CARLA_SAFE_ASSERT_CONTINUE(stateCustomData->isValid());
  615. MemoryOutputStream customDataXml;
  616. customDataXml << "\n";
  617. customDataXml << " <CustomData>\n";
  618. customDataXml << " <Type>" << xmlSafeString(stateCustomData->type, true) << "</Type>\n";
  619. customDataXml << " <Key>" << xmlSafeString(stateCustomData->key, true) << "</Key>\n";
  620. if (std::strcmp(stateCustomData->type, CUSTOM_DATA_TYPE_CHUNK) == 0 || std::strlen(stateCustomData->value) >= 128)
  621. {
  622. customDataXml << " <Value>\n";
  623. customDataXml << xmlSafeStringFast(stateCustomData->value, true);
  624. customDataXml << "\n </Value>\n";
  625. }
  626. else
  627. {
  628. customDataXml << " <Value>";
  629. customDataXml << xmlSafeStringFast(stateCustomData->value, true);
  630. customDataXml << "</Value>\n";
  631. }
  632. customDataXml << " </CustomData>\n";
  633. content << customDataXml;
  634. }
  635. if (chunk != nullptr && chunk[0] != '\0')
  636. {
  637. MemoryOutputStream chunkXml, chunkSplt;
  638. getNewLineSplittedString(chunkSplt, chunk);
  639. chunkXml << "\n <Chunk>\n";
  640. chunkXml << chunkSplt;
  641. chunkXml << "\n </Chunk>\n";
  642. content << chunkXml;
  643. }
  644. content << " </Data>\n";
  645. }
  646. // -----------------------------------------------------------------------
  647. CARLA_BACKEND_END_NAMESPACE