/* * Carla Plugin Host * Copyright (C) 2011-2019 Filipe Coelho * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as * published by the Free Software Foundation; either version 2 of * the License, or any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * For a full copy of the GNU General Public License see the doc/GPL.txt file. */ #include "CarlaEngineOsc.hpp" #ifdef HAVE_LIBLO #include "CarlaEngineInternal.hpp" #include "CarlaPlugin.hpp" #include "CarlaMIDI.h" #include CARLA_BACKEND_START_NAMESPACE // ----------------------------------------------------------------------- int CarlaEngineOsc::handleMessage(const bool isTCP, const char* const path, const int argc, const lo_arg* const* const argv, const char* const types, const lo_message msg) { CARLA_SAFE_ASSERT_RETURN(fName.isNotEmpty(), 1); CARLA_SAFE_ASSERT_RETURN(path != nullptr && path[0] != '\0', 1); CARLA_SAFE_ASSERT_RETURN(path[0] == '/', 1); #ifdef DEBUG if (std::strncmp(path, "/bridge_pong", 12) != 0) { carla_debug("CarlaEngineOsc::handleMessage(%s, \"%s\", %i, %p, \"%s\", %p)", bool2str(isTCP), path, argc, argv, types, msg); } #endif if (isTCP) { CARLA_SAFE_ASSERT_RETURN(fServerPathTCP.isNotEmpty(), 1); CARLA_SAFE_ASSERT_RETURN(fServerTCP != nullptr, 1); } else { CARLA_SAFE_ASSERT_RETURN(fServerPathUDP.isNotEmpty(), 1); CARLA_SAFE_ASSERT_RETURN(fServerUDP != nullptr, 1); } const lo_address source = lo_message_get_source(msg); // Initial path check if (std::strcmp(path, "/register") == 0) return handleMsgRegister(isTCP, argc, argv, types, source); if (std::strcmp(path, "/unregister") == 0) return handleMsgUnregister(isTCP, argc, argv, types); if (std::strncmp(path, "/ctrl/", 6) == 0) { CARLA_SAFE_ASSERT_RETURN(isTCP, 1); return handleMsgControl(path + 6, argc, argv, types); } // Check if message is for this client std::size_t bytesAfterName; if (fControlDataTCP.owner != nullptr && std::strcmp(lo_address_get_hostname(source), fControlDataTCP.owner) == 0) { if (const char* const slash = std::strchr(path+1, '/')) { bytesAfterName = static_cast(slash - path); } else { carla_stderr("CarlaEngineOsc::handleMessage() - message '%s' is invalid", path); return 1; } } else { bytesAfterName = fName.length(); if (std::strlen(path) <= bytesAfterName || std::strncmp(path+1, fName, bytesAfterName) != 0) { carla_stderr("CarlaEngineOsc::handleMessage() - message not for this client -> '%s' != '/%s/'", path, fName.buffer()); return 1; } ++bytesAfterName; } // Get plugin id from path, "/carla/23/method" -> 23 uint pluginId = 0; std::size_t offset; if (! std::isdigit(path[bytesAfterName+1])) { carla_stderr("CarlaEngineOsc::handleMessage() - invalid message '%s'", path); return 1; } if (std::isdigit(path[bytesAfterName+2])) { if (std::isdigit(path[bytesAfterName+4])) { carla_stderr2("CarlaEngineOsc::handleMessage() - invalid plugin id, over 999? (value: \"%s\")", path+bytesAfterName); return 1; } else if (std::isdigit(path[bytesAfterName+3])) { // 3 digits, /xyz/method offset = 5; pluginId += uint(path[bytesAfterName+1]-'0')*100; pluginId += uint(path[bytesAfterName+2]-'0')*10; pluginId += uint(path[bytesAfterName+3]-'0'); } else { // 2 digits, /xy/method offset = 4; pluginId += uint(path[bytesAfterName+1]-'0')*10; pluginId += uint(path[bytesAfterName+2]-'0'); } } else { // single digit, /x/method offset = 3; pluginId += uint(path[bytesAfterName+1]-'0'); } if (pluginId > fEngine->getCurrentPluginCount()) { carla_stderr("CarlaEngineOsc::handleMessage() - failed to get plugin, wrong id '%i'", pluginId); return 0; } // Get plugin const CarlaPluginPtr plugin = fEngine->getPluginUnchecked(pluginId); if (plugin == nullptr || plugin->getId() != pluginId) { carla_stderr("CarlaEngineOsc::handleMessage() - invalid plugin id '%i', probably has been removed (path: '%s')", pluginId, path); return 0; } // Get method from path, "/Carla/i/method" -> "method" char method[48]; std::strncpy(method, path + (bytesAfterName + offset), 47); method[47] = '\0'; if (method[0] == '\0') { carla_stderr("CarlaEngineOsc::handleMessage(%s, \"%s\", ...) - received message without method", bool2str(isTCP), path); return 0; } // Internal methods if (std::strcmp(method, "set_option") == 0) return 0; //handleMsgSetOption(plugin, argc, argv, types); // TODO if (std::strcmp(method, "set_active") == 0) return handleMsgSetActive(plugin, argc, argv, types); if (std::strcmp(method, "set_drywet") == 0) return handleMsgSetDryWet(plugin, argc, argv, types); if (std::strcmp(method, "set_volume") == 0) return handleMsgSetVolume(plugin, argc, argv, types); if (std::strcmp(method, "set_balance_left") == 0) return handleMsgSetBalanceLeft(plugin, argc, argv, types); if (std::strcmp(method, "set_balance_right") == 0) return handleMsgSetBalanceRight(plugin, argc, argv, types); if (std::strcmp(method, "set_panning") == 0) return handleMsgSetPanning(plugin, argc, argv, types); if (std::strcmp(method, "set_ctrl_channel") == 0) return 0; //handleMsgSetControlChannel(plugin, argc, argv, types); // TODO if (std::strcmp(method, "set_parameter_value") == 0) return handleMsgSetParameterValue(plugin, argc, argv, types); if (std::strcmp(method, "set_parameter_mapped_control_index") == 0) return handleMsgSetParameterMappedControlIndex(plugin, argc, argv, types); if (std::strcmp(method, "set_parameter_mapped_range") == 0) return handleMsgSetParameterMappedRange(plugin, argc, argv, types); if (std::strcmp(method, "set_parameter_midi_channel") == 0) return handleMsgSetParameterMidiChannel(plugin, argc, argv, types); if (std::strcmp(method, "set_program") == 0) return handleMsgSetProgram(plugin, argc, argv, types); if (std::strcmp(method, "set_midi_program") == 0) return handleMsgSetMidiProgram(plugin, argc, argv, types); if (std::strcmp(method, "set_custom_data") == 0) return 0; //handleMsgSetCustomData(plugin, argc, argv, types); // TODO if (std::strcmp(method, "set_chunk") == 0) return 0; //handleMsgSetChunk(plugin, argc, argv, types); // TODO if (std::strcmp(method, "note_on") == 0) return handleMsgNoteOn(plugin, argc, argv, types); if (std::strcmp(method, "note_off") == 0) return handleMsgNoteOff(plugin, argc, argv, types); // Send all other methods to plugins, TODO plugin->handleOscMessage(method, argc, argv, types, msg); return 0; } // ----------------------------------------------------------------------- int CarlaEngineOsc::handleMsgRegister(const bool isTCP, const int argc, const lo_arg* const* const argv, const char* const types, const lo_address source) { carla_debug("CarlaEngineOsc::handleMsgRegister()"); CARLA_ENGINE_OSC_CHECK_OSC_TYPES(1, "s"); const char* const url = &argv[0]->s; CarlaOscData& oscData(isTCP ? fControlDataTCP : fControlDataUDP); if (oscData.owner != nullptr) { carla_stderr("OSC backend already registered to %s", oscData.owner); char* const path = lo_url_get_path(url); const size_t pathlen = std::strlen(path); CARLA_SAFE_ASSERT_RETURN(pathlen < 32, 0); char targetPath[pathlen+12]; std::strcpy(targetPath, path); std::strcat(targetPath, "/exit-error"); lo_send_from(source, isTCP ? fServerTCP : fServerUDP, LO_TT_IMMEDIATE, targetPath, "s", "OSC already registered to another client"); free(path); } else { const char* const host = lo_address_get_hostname(source); /* */ char* const port = lo_url_get_port(url); // NOTE: lo_address_get_port is buggy against TCP const lo_address target = lo_address_new_with_proto(isTCP ? LO_TCP : LO_UDP, host, port); oscData.owner = carla_strdup_safe(host); oscData.path = carla_strdup_free(lo_url_get_path(url)); oscData.target = target; char* const targeturl = lo_address_get_url(target); carla_stdout("OSC %s backend registered to %s, path: %s, target: %s (host: %s, port: %s)", isTCP ? "TCP" : "UDP", url, oscData.path, targeturl, host, port); free(targeturl); free(port); if (isTCP) { const EngineOptions& opts(fEngine->getOptions()); fEngine->callback(false, true, ENGINE_CALLBACK_ENGINE_STARTED, fEngine->getCurrentPluginCount(), opts.processMode, opts.transportMode, static_cast(fEngine->getBufferSize()), static_cast(fEngine->getSampleRate()), fEngine->getCurrentDriverName()); for (uint i=0, count=fEngine->getCurrentPluginCount(); i < count; ++i) { const CarlaPluginPtr plugin = fEngine->getPluginUnchecked(i); CARLA_SAFE_ASSERT_CONTINUE(plugin != nullptr); fEngine->callback(false, true, ENGINE_CALLBACK_PLUGIN_ADDED, i, plugin->getType(), 0, 0, 0.0f, plugin->getName()); } fEngine->patchbayRefresh(false, true, fEngine->pData->graph.isUsingExternalOSC()); } } return 0; } int CarlaEngineOsc::handleMsgUnregister(const bool isTCP, const int argc, const lo_arg* const* const argv, const char* const types) { carla_debug("CarlaEngineOsc::handleMsgUnregister()"); CARLA_ENGINE_OSC_CHECK_OSC_TYPES(1, "s"); CarlaOscData& oscData(isTCP ? fControlDataTCP : fControlDataUDP); if (oscData.owner == nullptr) { carla_stderr("OSC backend is not registered yet, unregister failed"); return 0; } const char* const url = &argv[0]->s; if (std::strcmp(oscData.owner, url) == 0) { carla_stdout("OSC client %s unregistered", url); oscData.clear(); return 0; } carla_stderr("OSC backend unregister failed, current owner %s does not match requested %s", oscData.owner, url); return 0; } int CarlaEngineOsc::handleMsgControl(const char* const method, const int argc, const lo_arg* const* const argv, const char* const types) { carla_debug("CarlaEngineOsc::handleMsgControl()"); CARLA_SAFE_ASSERT_RETURN(method != nullptr && method[0] != '\0', 0); CARLA_SAFE_ASSERT_RETURN(types != nullptr, 0); CARLA_SAFE_ASSERT_RETURN(types[0] == 'i', 0); if (fControlDataTCP.owner == nullptr) { carla_stderr("OSC backend is not registered yet, control failed"); return 0; } const int32_t messageId = argv[0]->i; bool ok; #define CARLA_SAFE_ASSERT_RETURN_OSC_ERR(cond) \ if (! (cond)) { carla_safe_assert(#cond, __FILE__, __LINE__); sendResponse(messageId, #cond); return 0; } /**/ if (std::strcmp(method, "clear_engine_xruns") == 0) { ok = true; fEngine->clearXruns(); } else if (std::strcmp(method, "cancel_engine_action") == 0) { ok = true; fEngine->setActionCanceled(true); } else if (std::strcmp(method, "patchbay_connect") == 0) { CARLA_SAFE_ASSERT_RETURN_OSC_ERR(argc == 6); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[1] == 'i'); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[2] == 'i'); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[3] == 'i'); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[4] == 'i'); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[5] == 'i'); const bool external = argv[1]->i != 0; const int32_t groupA = argv[2]->i; CARLA_SAFE_ASSERT_RETURN_OSC_ERR(groupA >= 0); const int32_t portA = argv[3]->i; CARLA_SAFE_ASSERT_RETURN_OSC_ERR(portA >= 0); const int32_t groupB = argv[4]->i; CARLA_SAFE_ASSERT_RETURN_OSC_ERR(groupB >= 0); const int32_t portB = argv[5]->i; CARLA_SAFE_ASSERT_RETURN_OSC_ERR(portB >= 0); ok = fEngine->patchbayConnect(external, static_cast(groupA), static_cast(portA), static_cast(groupB), static_cast(portB)); } else if (std::strcmp(method, "patchbay_disconnect") == 0) { CARLA_SAFE_ASSERT_RETURN_OSC_ERR(argc == 3); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[1] == 'i'); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[2] == 'i'); const bool external = argv[1]->i != 0; const int32_t connectionId = argv[2]->i; CARLA_SAFE_ASSERT_RETURN_OSC_ERR(connectionId >= 0); ok = fEngine->patchbayDisconnect(external, static_cast(connectionId)); } else if (std::strcmp(method, "patchbay_set_group_pos") == 0) { CARLA_SAFE_ASSERT_RETURN_OSC_ERR(argc == 7); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[1] == 'i'); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[2] == 'i'); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[3] == 'i'); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[4] == 'i'); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[5] == 'i'); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[6] == 'i'); const bool external = argv[1]->i != 0; const int32_t groupId = argv[2]->i; CARLA_SAFE_ASSERT_RETURN_OSC_ERR(groupId >= 0); ok = fEngine->patchbaySetGroupPos(true, false, external, static_cast(groupId), argv[3]->i, argv[4]->i, argv[5]->i, argv[6]->i); } else if (std::strcmp(method, "patchbay_refresh") == 0) { CARLA_SAFE_ASSERT_RETURN_OSC_ERR(argc == 2); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[1] == 'i'); const bool external = argv[1]->i != 0; ok = fEngine->patchbayRefresh(false, true, external); } else if (std::strcmp(method, "transport_play") == 0) { ok = true; fEngine->transportPlay(); } else if (std::strcmp(method, "transport_pause") == 0) { ok = true; fEngine->transportPause(); } else if (std::strcmp(method, "transport_bpm") == 0) { CARLA_SAFE_ASSERT_RETURN_OSC_ERR(argc == 2); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[1] == 'f'); const double bpm = argv[1]->f; CARLA_SAFE_ASSERT_RETURN_OSC_ERR(bpm >= 0.0); ok = true; fEngine->transportBPM(bpm); } else if (std::strcmp(method, "transport_relocate") == 0) { CARLA_SAFE_ASSERT_RETURN_OSC_ERR(argc == 2); uint64_t frame; /**/ if (types[1] == 'i') { const int32_t i = argv[1]->i; CARLA_SAFE_ASSERT_RETURN_OSC_ERR(i >= 0); frame = static_cast(i); } else if (types[1] == 'h') { const int64_t h = argv[1]->h; CARLA_SAFE_ASSERT_RETURN_OSC_ERR(h >= 0); frame = static_cast(h); } else { carla_stderr2("Wrong OSC type used for '%s'", method); sendResponse(messageId, "Wrong OSC type"); return 0; } ok = true; fEngine->transportRelocate(frame); } else if (std::strcmp(method, "add_plugin") == 0) { CARLA_SAFE_ASSERT_RETURN_OSC_ERR(argc == 8); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[1] == 'i'); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[2] == 'i'); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[3] == 's'); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[4] == 's'); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[5] == 's'); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[7] == 'i'); int32_t btype = argv[1]->i; CARLA_SAFE_ASSERT_RETURN_OSC_ERR(btype >= 0); const int32_t ptype = argv[2]->i; CARLA_SAFE_ASSERT_RETURN_OSC_ERR(ptype >= 0); // Force binary type to be native in some cases switch (ptype) { case PLUGIN_INTERNAL: case PLUGIN_LV2: case PLUGIN_SF2: case PLUGIN_SFZ: case PLUGIN_JACK: btype = BINARY_NATIVE; break; } const char* filename = &argv[3]->s; if (filename != nullptr && std::strcmp(filename, "(null)") == 0) filename = nullptr; const char* name = &argv[4]->s; if (name != nullptr && std::strcmp(name, "(null)") == 0) name = nullptr; const char* const label = &argv[5]->s; CARLA_SAFE_ASSERT_RETURN_OSC_ERR(label != nullptr && label[0] != '\0'); int64_t uniqueId; /**/ if (types[6] == 'i') { uniqueId = argv[6]->i; } else if (types[6] == 'h') { uniqueId = argv[6]->h; } else { carla_stderr2("Wrong OSC type used for '%s' uniqueId", method); sendResponse(messageId, "Wrong OSC type"); return 0; } const int32_t options = argv[7]->i; CARLA_SAFE_ASSERT_RETURN_OSC_ERR(options >= 0); ok = fEngine->addPlugin(static_cast(btype), static_cast(ptype), filename, name, label, uniqueId, nullptr, static_cast(options)); } else if (std::strcmp(method, "remove_plugin") == 0) { CARLA_SAFE_ASSERT_RETURN_OSC_ERR(argc == 2); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[1] == 'i'); const int32_t id = argv[1]->i; CARLA_SAFE_ASSERT_RETURN_OSC_ERR(id >= 0); ok = fEngine->removePlugin(static_cast(id)); } else if (std::strcmp(method, "remove_all_plugins") == 0) { ok = fEngine->removeAllPlugins(); } else if (std::strcmp(method, "rename_plugin") == 0) { CARLA_SAFE_ASSERT_RETURN_OSC_ERR(argc == 3); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[1] == 'i'); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[2] == 's'); const int32_t id = argv[1]->i; CARLA_SAFE_ASSERT_RETURN_OSC_ERR(id >= 0); const char* const newName = &argv[2]->s; CARLA_SAFE_ASSERT_RETURN_OSC_ERR(newName != nullptr && newName[0] != '\0'); ok = fEngine->renamePlugin(static_cast(id), newName); } else if (std::strcmp(method, "clone_plugin") == 0) { CARLA_SAFE_ASSERT_RETURN_OSC_ERR(argc == 2); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[1] == 'i'); const int32_t id = argv[1]->i; CARLA_SAFE_ASSERT_RETURN_OSC_ERR(id >= 0); ok = fEngine->clonePlugin(static_cast(id)); } else if (std::strcmp(method, "replace_plugin") == 0) { CARLA_SAFE_ASSERT_RETURN_OSC_ERR(argc == 2); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[1] == 'i'); const int32_t id = argv[1]->i; CARLA_SAFE_ASSERT_RETURN_OSC_ERR(id >= 0); ok = fEngine->replacePlugin(static_cast(id)); } else if (std::strcmp(method, "switch_plugins") == 0) { CARLA_SAFE_ASSERT_RETURN_OSC_ERR(argc == 3); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[1] == 'i'); CARLA_SAFE_ASSERT_RETURN_OSC_ERR(types[2] == 'i'); const int32_t idA = argv[1]->i; CARLA_SAFE_ASSERT_RETURN_OSC_ERR(idA >= 0); const int32_t idB = argv[2]->i; CARLA_SAFE_ASSERT_RETURN_OSC_ERR(idB >= 0); ok = fEngine->switchPlugins(static_cast(idA), static_cast(idB)); } else { carla_stderr2("Unhandled OSC control for '%s'", method); sendResponse(messageId, "Unhandled OSC control method"); return 0; } #undef CARLA_SAFE_ASSERT_RETURN_OSC_ERR sendResponse(messageId, ok ? "" : fEngine->getLastError()); return 0; } // ----------------------------------------------------------------------- int CarlaEngineOsc::handleMsgSetActive(CARLA_ENGINE_OSC_HANDLE_ARGS) { carla_debug("CarlaEngineOsc::handleMsgSetActive()"); CARLA_ENGINE_OSC_CHECK_OSC_TYPES(1, "i"); const bool active = (argv[0]->i != 0); plugin->setActive(active, false, true); return 0; } int CarlaEngineOsc::handleMsgSetDryWet(CARLA_ENGINE_OSC_HANDLE_ARGS) { carla_debug("CarlaEngineOsc::handleMsgSetDryWet()"); CARLA_ENGINE_OSC_CHECK_OSC_TYPES(1, "f"); const float value = argv[0]->f; plugin->setDryWet(value, false, true); return 0; } int CarlaEngineOsc::handleMsgSetVolume(CARLA_ENGINE_OSC_HANDLE_ARGS) { carla_debug("CarlaEngineOsc::handleMsgSetVolume()"); CARLA_ENGINE_OSC_CHECK_OSC_TYPES(1, "f"); const float value = argv[0]->f; plugin->setVolume(value, false, true); return 0; } int CarlaEngineOsc::handleMsgSetBalanceLeft(CARLA_ENGINE_OSC_HANDLE_ARGS) { carla_debug("CarlaEngineOsc::handleMsgSetBalanceLeft()"); CARLA_ENGINE_OSC_CHECK_OSC_TYPES(1, "f"); const float value = argv[0]->f; plugin->setBalanceLeft(value, false, true); return 0; } int CarlaEngineOsc::handleMsgSetBalanceRight(CARLA_ENGINE_OSC_HANDLE_ARGS) { carla_debug("CarlaEngineOsc::handleMsgSetBalanceRight()"); CARLA_ENGINE_OSC_CHECK_OSC_TYPES(1, "f"); const float value = argv[0]->f; plugin->setBalanceRight(value, false, true); return 0; } int CarlaEngineOsc::handleMsgSetPanning(CARLA_ENGINE_OSC_HANDLE_ARGS) { carla_debug("CarlaEngineOsc::handleMsgSetPanning()"); CARLA_ENGINE_OSC_CHECK_OSC_TYPES(1, "f"); const float value = argv[0]->f; plugin->setPanning(value, false, true); return 0; } int CarlaEngineOsc::handleMsgSetParameterValue(CARLA_ENGINE_OSC_HANDLE_ARGS) { carla_debug("CarlaEngineOsc::handleMsgSetParameterValue()"); CARLA_ENGINE_OSC_CHECK_OSC_TYPES(2, "if"); const int32_t index = argv[0]->i; const float value = argv[1]->f; CARLA_SAFE_ASSERT_RETURN(index >= 0, 0); plugin->setParameterValue(static_cast(index), value, true, false, true); return 0; } int CarlaEngineOsc::handleMsgSetParameterMappedControlIndex(CARLA_ENGINE_OSC_HANDLE_ARGS) { carla_debug("CarlaEngineOsc::handleMsgSetParameterMappedIndex()"); CARLA_ENGINE_OSC_CHECK_OSC_TYPES(2, "ii"); const int32_t index = argv[0]->i; const int32_t ctrl = argv[1]->i; CARLA_SAFE_ASSERT_RETURN(index >= 0, 0); CARLA_SAFE_ASSERT_RETURN(ctrl >= CONTROL_INDEX_NONE && ctrl <= CONTROL_INDEX_MAX_ALLOWED, 0); plugin->setParameterMappedControlIndex(static_cast(index), static_cast(ctrl), false, true, true); return 0; } int CarlaEngineOsc::handleMsgSetParameterMappedRange(CARLA_ENGINE_OSC_HANDLE_ARGS) { carla_debug("CarlaEngineOsc::handleMsgSetParameterMappedRange()"); CARLA_ENGINE_OSC_CHECK_OSC_TYPES(2, "iff"); const int32_t index = argv[0]->i; const float minimum = argv[1]->f; const float maximum = argv[2]->f; CARLA_SAFE_ASSERT_RETURN(index >= 0, 0); plugin->setParameterMappedRange(static_cast(index), minimum, maximum, false, true); return 0; } int CarlaEngineOsc::handleMsgSetParameterMidiChannel(CARLA_ENGINE_OSC_HANDLE_ARGS) { carla_debug("CarlaEngineOsc::handleMsgSetParameterMidiChannel()"); CARLA_ENGINE_OSC_CHECK_OSC_TYPES(2, "ii"); const int32_t index = argv[0]->i; const int32_t channel = argv[1]->i; CARLA_SAFE_ASSERT_RETURN(index >= 0, 0); CARLA_SAFE_ASSERT_RETURN(channel >= 0 && channel < MAX_MIDI_CHANNELS, 0); plugin->setParameterMidiChannel(static_cast(index), static_cast(channel), false, true); return 0; } int CarlaEngineOsc::handleMsgSetProgram(CARLA_ENGINE_OSC_HANDLE_ARGS) { carla_debug("CarlaEngineOsc::handleMsgSetProgram()"); CARLA_ENGINE_OSC_CHECK_OSC_TYPES(1, "i"); const int32_t index = argv[0]->i; CARLA_SAFE_ASSERT_RETURN(index >= -1, 0); plugin->setProgram(index, true, false, true); return 0; } int CarlaEngineOsc::handleMsgSetMidiProgram(CARLA_ENGINE_OSC_HANDLE_ARGS) { carla_debug("CarlaEngineOsc::handleMsgSetMidiProgram()"); CARLA_ENGINE_OSC_CHECK_OSC_TYPES(1, "i"); const int32_t index = argv[0]->i; CARLA_SAFE_ASSERT_RETURN(index >= -1, 0); plugin->setMidiProgram(index, true, false, true); return 0; } int CarlaEngineOsc::handleMsgNoteOn(CARLA_ENGINE_OSC_HANDLE_ARGS) { carla_debug("CarlaEngineOsc::handleMsgNoteOn()"); CARLA_ENGINE_OSC_CHECK_OSC_TYPES(3, "iii"); const int32_t channel = argv[0]->i; const int32_t note = argv[1]->i; const int32_t velo = argv[2]->i; CARLA_SAFE_ASSERT_RETURN(channel >= 0 && channel < MAX_MIDI_CHANNELS, 0); CARLA_SAFE_ASSERT_RETURN(note >= 0 && note < MAX_MIDI_NOTE, 0); CARLA_SAFE_ASSERT_RETURN(velo >= 0 && velo < MAX_MIDI_VALUE, 0); plugin->sendMidiSingleNote(static_cast(channel), static_cast(note), static_cast(velo), true, false, true); return 0; } int CarlaEngineOsc::handleMsgNoteOff(CARLA_ENGINE_OSC_HANDLE_ARGS) { carla_debug("CarlaEngineOsc::handleMsgNoteOff()"); CARLA_ENGINE_OSC_CHECK_OSC_TYPES(2, "ii"); const int32_t channel = argv[0]->i; const int32_t note = argv[1]->i; CARLA_SAFE_ASSERT_RETURN(channel >= 0 && channel < MAX_MIDI_CHANNELS, 0); CARLA_SAFE_ASSERT_RETURN(note >= 0 && note < MAX_MIDI_NOTE, 0); plugin->sendMidiSingleNote(static_cast(channel), static_cast(note), 0, true, false, true); return 0; } // ----------------------------------------------------------------------- CARLA_BACKEND_END_NAMESPACE #endif // HAVE_LIBLO