@@ -899,16 +899,6 @@ CARLA_EXPORT const char* carla_get_library_filename(); | |||
*/ | |||
CARLA_EXPORT const char* carla_get_library_folder(); | |||
/*! | |||
* Initialize NSM (announce). | |||
*/ | |||
CARLA_EXPORT bool carla_nsm_init(int pid, const char* executableName); | |||
/*! | |||
* Allow NSM callbacks. | |||
*/ | |||
CARLA_EXPORT void carla_nsm_ready(int action); | |||
/** @} */ | |||
#endif /* CARLA_HOST_H_INCLUDED */ |
@@ -16,6 +16,7 @@ | |||
*/ | |||
#include "CarlaHost.h" | |||
#include "CarlaUtils.hpp" | |||
#include "juce_core.h" | |||
@@ -105,7 +106,7 @@ const char* carla_get_library_filename() | |||
if (ret.isEmpty()) | |||
{ | |||
using juce::File; | |||
using namespace juce; | |||
ret = File(File::getSpecialLocation(File::currentExecutableFile)).getFullPathName().toRawUTF8(); | |||
} | |||
@@ -120,7 +121,7 @@ const char* carla_get_library_folder() | |||
if (ret.isEmpty()) | |||
{ | |||
using juce::File; | |||
using namespace juce; | |||
ret = File(File::getSpecialLocation(File::currentExecutableFile).getParentDirectory()).getFullPathName().toRawUTF8(); | |||
} | |||
@@ -28,7 +28,6 @@ | |||
#include "CarlaBase64Utils.hpp" | |||
#ifdef HAVE_LIBLO | |||
# include "CarlaOscUtils.hpp" | |||
#endif | |||
#include "juce_audio_formats.h" | |||
@@ -80,656 +79,7 @@ struct CarlaBackendStandalone { | |||
CARLA_DECLARE_NON_COPY_STRUCT(CarlaBackendStandalone) | |||
}; | |||
static CarlaBackendStandalone gStandalone; | |||
#ifdef HAVE_LIBLO | |||
// ------------------------------------------------------------------------------------------------------------------- | |||
// NSM support | |||
#define NSM_API_VERSION_MAJOR 1 | |||
#define NSM_API_VERSION_MINOR 2 | |||
#define NSM_CLIENT_FEATURES ":switch:" | |||
//#define NSM_CLIENT_FEATURES ":switch:optional-gui:" | |||
class CarlaNSM | |||
{ | |||
public: | |||
CarlaNSM() noexcept | |||
: fOscAddress(nullptr), | |||
fOscServer(nullptr), | |||
fOscServerThread(nullptr), | |||
fClientNameId(), | |||
fProjectPath(), | |||
fHasBroadcast(false), | |||
fHasOptionalGui(false), | |||
fHasServerControl(false), | |||
fStarted(), | |||
fReadyActionOpen(true), | |||
fReadyActionSave(true) {} | |||
~CarlaNSM() | |||
{ | |||
CARLA_SAFE_ASSERT(fReadyActionOpen); | |||
CARLA_SAFE_ASSERT(fReadyActionSave); | |||
if (fOscServerThread != nullptr) | |||
{ | |||
lo_server_thread_stop(fOscServerThread); | |||
lo_server_thread_free(fOscServerThread); | |||
fOscServerThread = nullptr; | |||
fOscServer = nullptr; | |||
} | |||
if (fOscAddress != nullptr) | |||
{ | |||
lo_address_free(fOscAddress); | |||
fOscAddress = nullptr; | |||
} | |||
} | |||
bool announce(const int pid, const char* const executableName) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(pid != 0, false); | |||
CARLA_SAFE_ASSERT_RETURN(executableName != nullptr && executableName[0] != '\0', false); | |||
const char* const NSM_URL(std::getenv("NSM_URL")); | |||
if (NSM_URL == nullptr) | |||
return false; | |||
const lo_address addr = lo_address_new_from_url(NSM_URL); | |||
CARLA_SAFE_ASSERT_RETURN(addr != nullptr, false); | |||
const int proto = lo_address_get_protocol(addr); | |||
if (fOscServerThread == nullptr) | |||
{ | |||
// create new OSC server | |||
fOscServerThread = lo_server_thread_new_with_proto(nullptr, proto, _osc_error_handler); | |||
CARLA_SAFE_ASSERT_RETURN(fOscServerThread != nullptr, false); | |||
// register message handlers | |||
lo_server_thread_add_method(fOscServerThread, "/error", "sis", _error_handler, this); | |||
lo_server_thread_add_method(fOscServerThread, "/reply", "ssss", _reply_handler, this); | |||
lo_server_thread_add_method(fOscServerThread, "/nsm/client/open", "sss", _open_handler, this); | |||
lo_server_thread_add_method(fOscServerThread, "/nsm/client/save", "", _save_handler, this); | |||
lo_server_thread_add_method(fOscServerThread, "/nsm/client/session_is_loaded", "", _loaded_handler, this); | |||
lo_server_thread_add_method(fOscServerThread, "/nsm/client/show_optional_gui", "", _show_gui_handler, this); | |||
lo_server_thread_add_method(fOscServerThread, "/nsm/client/hide_optional_gui", "", _hide_gui_handler, this); | |||
lo_server_thread_add_method(fOscServerThread, nullptr, nullptr, _broadcast_handler, this); | |||
fOscServer = lo_server_thread_get_server(fOscServerThread); | |||
} | |||
#ifndef BUILD_ANSI_TEST | |||
lo_send_from(addr, fOscServer, LO_TT_IMMEDIATE, "/nsm/server/announce", "sssiii", | |||
"Carla", NSM_CLIENT_FEATURES, executableName, NSM_API_VERSION_MAJOR, NSM_API_VERSION_MINOR, pid); | |||
#endif | |||
lo_address_free(addr); | |||
return true; | |||
} | |||
void ready(const int action) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(fOscServerThread != nullptr,); | |||
switch (action) | |||
{ | |||
case -1: // init | |||
CARLA_SAFE_ASSERT_BREAK(! fStarted); | |||
fStarted = true; | |||
lo_server_thread_start(fOscServerThread); | |||
break; | |||
case 0: // error | |||
break; | |||
case 1: // reply | |||
break; | |||
case 2: // open | |||
fReadyActionOpen = true; | |||
break; | |||
case 3: // save | |||
fReadyActionSave = true; | |||
break; | |||
case 4: // session loaded | |||
break; | |||
case 5: // show gui | |||
CARLA_SAFE_ASSERT_BREAK(fOscAddress != nullptr); | |||
CARLA_SAFE_ASSERT_BREAK(fOscServer != nullptr); | |||
lo_send_from(fOscAddress, fOscServer, LO_TT_IMMEDIATE, "/nsm/client/gui_is_shown", ""); | |||
break; | |||
case 6: // hide gui | |||
CARLA_SAFE_ASSERT_BREAK(fOscAddress != nullptr); | |||
CARLA_SAFE_ASSERT_BREAK(fOscServer != nullptr); | |||
lo_send_from(fOscAddress, fOscServer, LO_TT_IMMEDIATE, "/nsm/client/gui_is_hidden", ""); | |||
break; | |||
} | |||
} | |||
static CarlaNSM& getInstance() | |||
{ | |||
static CarlaNSM sInstance; | |||
return sInstance; | |||
} | |||
protected: | |||
int handleError(const char* const method, const int code, const char* const message) | |||
{ | |||
carla_stdout("CarlaNSM::handleError(\"%s\", %i, \"%s\")", method, code, message); | |||
if (gStandalone.engineCallback != nullptr) | |||
gStandalone.engineCallback(gStandalone.engineCallbackPtr, CB::ENGINE_CALLBACK_NSM, 0, 0, code, 0.0f, message); | |||
return 1; | |||
// may be unused | |||
(void)method; | |||
} | |||
int handleReply(const char* const method, const char* const message, const char* const smName, const char* const features, | |||
const lo_message msg) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(fOscServerThread != nullptr, 1); | |||
carla_stdout("CarlaNSM::handleReply(\"%s\", \"%s\", \"%s\", \"%s\")", method, message, smName, features); | |||
if (std::strcmp(method, "/nsm/server/announce") == 0) | |||
{ | |||
char* const addressURL(lo_address_get_url(lo_message_get_source(msg))); | |||
CARLA_SAFE_ASSERT_RETURN(addressURL != nullptr, 0); | |||
if (fOscAddress != nullptr) | |||
lo_address_free(fOscAddress); | |||
fOscAddress = lo_address_new_from_url(addressURL); | |||
CARLA_SAFE_ASSERT_RETURN(fOscAddress != nullptr, 0); | |||
fHasBroadcast = std::strstr(features, ":broadcast:") != nullptr; | |||
fHasOptionalGui = std::strstr(features, ":optional-gui:") != nullptr; | |||
fHasServerControl = std::strstr(features, ":server_control:") != nullptr; | |||
// UI starts visible | |||
if (fHasOptionalGui) | |||
lo_send_from(fOscAddress, fOscServer, LO_TT_IMMEDIATE, "/nsm/client/gui_is_shown", ""); | |||
carla_stdout("Carla started via '%s', message: %s", smName, message); | |||
if (gStandalone.engineCallback != nullptr) | |||
{ | |||
int flags = 0; | |||
if (fHasBroadcast) | |||
flags |= 1 << 0; | |||
if (fHasOptionalGui) | |||
flags |= 1 << 1; | |||
if (fHasServerControl) | |||
flags |= 1 << 2; | |||
gStandalone.engineCallback(gStandalone.engineCallbackPtr, CB::ENGINE_CALLBACK_NSM, 0, 1, flags, 0.0f, smName); | |||
} | |||
} | |||
else | |||
{ | |||
carla_stdout("Got unknown NSM reply method '%s'", method); | |||
} | |||
return 0; | |||
} | |||
int handleOpen(const char* const projectPath, const char* const displayName, const char* const clientNameId) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(fOscAddress != nullptr, 1); | |||
CARLA_SAFE_ASSERT_RETURN(fOscServer != nullptr, 1); | |||
carla_stdout("CarlaNSM::handleOpen(\"%s\", \"%s\", \"%s\")", projectPath, displayName, clientNameId); | |||
if (gStandalone.engineCallback != nullptr) | |||
{ | |||
fReadyActionOpen = false; | |||
gStandalone.engineCallback(gStandalone.engineCallbackPtr, CB::ENGINE_CALLBACK_NSM, 0, 2, 0, 0.0f, projectPath); | |||
for (; ! fReadyActionOpen;) | |||
carla_msleep(10); | |||
} | |||
else | |||
{ | |||
using namespace juce; | |||
if (carla_is_engine_running()) | |||
carla_engine_close(); | |||
carla_engine_init("JACK", clientNameId); | |||
fProjectPath = projectPath; | |||
fProjectPath += ".carxp"; | |||
const String jfilename = String(CharPointer_UTF8(fProjectPath)); | |||
if (File(jfilename).existsAsFile()) | |||
carla_load_project(fProjectPath); | |||
} | |||
fClientNameId = clientNameId; | |||
lo_send_from(fOscAddress, fOscServer, LO_TT_IMMEDIATE, "/reply", "ss", "/nsm/client/open", "OK"); | |||
// Broadcast ourselves | |||
if (fHasBroadcast) | |||
{ | |||
char* const oscServerAddress(lo_server_get_url(fOscServer)); | |||
if (lo_message msg = lo_message_new()) | |||
{ | |||
lo_message_add(msg, "sssss", | |||
"/non/hello", | |||
oscServerAddress, | |||
"Carla", | |||
CARLA_VERSION_STRING, | |||
fClientNameId.buffer()); | |||
lo_send_message_from(fOscAddress, fOscServer, "/nsm/server/broadcast", msg); | |||
lo_message_free(msg); | |||
} | |||
std::free(oscServerAddress); | |||
} | |||
return 0; | |||
} | |||
int handleSave() | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(fOscAddress != nullptr, 1); | |||
CARLA_SAFE_ASSERT_RETURN(fOscServer != nullptr, 1); | |||
carla_stdout("CarlaNSM::handleSave()"); | |||
if (gStandalone.engineCallback != nullptr) | |||
{ | |||
fReadyActionSave = false; | |||
gStandalone.engineCallback(gStandalone.engineCallbackPtr, CB::ENGINE_CALLBACK_NSM, 0, 3, 0, 0.0f, nullptr); | |||
for (; ! fReadyActionSave;) | |||
carla_msleep(10); | |||
} | |||
else | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(fProjectPath.isNotEmpty(), 0); | |||
carla_save_project(fProjectPath); | |||
} | |||
lo_send_from(fOscAddress, fOscServer, LO_TT_IMMEDIATE, "/reply", "ss", "/nsm/client/save", "OK"); | |||
return 0; | |||
} | |||
int handleSessionIsLoaded() | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(fOscAddress != nullptr, 1); | |||
CARLA_SAFE_ASSERT_RETURN(fOscServer != nullptr, 1); | |||
carla_stdout("CarlaNSM::handleSessionIsLoaded()"); | |||
if (gStandalone.engineCallback != nullptr) | |||
gStandalone.engineCallback(gStandalone.engineCallbackPtr, CB::ENGINE_CALLBACK_NSM, 0, 4, 0, 0.0f, nullptr); | |||
return 0; | |||
} | |||
int handleShowOptionalGui() | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(fOscAddress != nullptr, 1); | |||
CARLA_SAFE_ASSERT_RETURN(fOscServer != nullptr, 1); | |||
carla_stdout("CarlaNSM::handleShowOptionalGui()"); | |||
if (gStandalone.engineCallback != nullptr) | |||
gStandalone.engineCallback(gStandalone.engineCallbackPtr, CB::ENGINE_CALLBACK_NSM, 0, 5, 0, 0.0f, nullptr); | |||
return 0; | |||
} | |||
int handleHideOptionalGui() | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(fOscAddress != nullptr, 1); | |||
CARLA_SAFE_ASSERT_RETURN(fOscServer != nullptr, 1); | |||
carla_stdout("CarlaNSM::handleHideOptionalGui()"); | |||
if (gStandalone.engineCallback != nullptr) | |||
gStandalone.engineCallback(gStandalone.engineCallbackPtr, CB::ENGINE_CALLBACK_NSM, 0, 6, 0, 0.0f, nullptr); | |||
return 0; | |||
} | |||
int handleBroadcast(const char* const path, const char* const types, lo_arg** const argv, const int argc, | |||
const lo_message msg) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(fOscAddress != nullptr, 1); | |||
CARLA_SAFE_ASSERT_RETURN(fOscServer != nullptr, 1); | |||
CARLA_SAFE_ASSERT_RETURN(argc >= 0, 0); | |||
carla_stdout("CarlaNSM::handleBroadcast(%s, %s, %p, %i)", path, types, argv, argc); | |||
if (std::strcmp(path, "/non/hello") == 0) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(argc == 4, 0); | |||
CARLA_SAFE_ASSERT_RETURN(std::strcmp(types, "ssss") == 0, 0); | |||
const char* const url = &argv[0]->s; | |||
//const char* const name = &argv[1]->s; | |||
//const char* const version = &argv[2]->s; | |||
//const char* const clientId = &argv[3]->s; | |||
const lo_address targetAddress(lo_address_new_from_url(url)); | |||
CARLA_SAFE_ASSERT_RETURN(targetAddress != nullptr, 0); | |||
char* const oscServerAddress(lo_server_get_url(fOscServer)); | |||
CARLA_SAFE_ASSERT_RETURN(oscServerAddress != nullptr, 0); | |||
if (lo_message msg2 = lo_message_new()) | |||
{ | |||
lo_message_add(msg2, "ss", | |||
fClientNameId.buffer(), | |||
oscServerAddress); | |||
lo_send_message_from(targetAddress, fOscServer, "/signal/hello", msg2); | |||
lo_message_free(msg2); | |||
carla_stdout("CarlaNSM::handleBroadcast - sent hello"); | |||
} | |||
lo_address_free(targetAddress); | |||
std::free(oscServerAddress); | |||
return 0; | |||
} | |||
if (std::strcmp(path, "/signal/hello") == 0) | |||
{ | |||
carla_stdout("CarlaNSM::handleBroadcast - got hello"); | |||
//const char* const name = &argv[0]->s; | |||
const char* const url = &argv[1]->s; | |||
const lo_address targetAddress(lo_address_new_from_url(url)); | |||
CARLA_SAFE_ASSERT_RETURN(targetAddress != nullptr, 0); | |||
char* const oscServerAddress(lo_server_get_url(fOscServer)); | |||
CARLA_SAFE_ASSERT_RETURN(oscServerAddress != nullptr, 0); | |||
//lo_send_from(targetAddress, fOscServer, LO_TT_IMMEDIATE, "/signal/list", ""); | |||
if (lo_message msg2 = lo_message_new()) | |||
{ | |||
lo_message_add(msg2, "ss", | |||
fClientNameId.buffer(), | |||
oscServerAddress); | |||
lo_send_message_from(targetAddress, fOscServer, "/signal/hello", msg2); | |||
lo_message_free(msg2); | |||
carla_stdout("CarlaNSM::handleBroadcast - sent hello"); | |||
} | |||
lo_address_free(targetAddress); | |||
std::free(oscServerAddress); | |||
return 0; | |||
} | |||
if (std::strcmp(path, "/signal/list") == 0) | |||
{ | |||
carla_stdout("CarlaNSM::handleBroadcast - got list"); | |||
CARLA_SAFE_ASSERT_RETURN(carla_is_engine_running(), 0); | |||
//const char* prefix = nullptr; | |||
//if (argc > 0) | |||
//prefix = &argv[0]->s; | |||
const lo_address targetAddress(lo_message_get_source(msg)); | |||
CARLA_SAFE_ASSERT_RETURN(targetAddress != nullptr, 0); | |||
CarlaString stripName; | |||
for (uint32_t i = 0, pluginCount = carla_get_current_plugin_count(); i < pluginCount; ++i) | |||
{ | |||
const CarlaPluginInfo* const pluginInfo(carla_get_plugin_info(i)); | |||
CARLA_SAFE_ASSERT_CONTINUE(pluginInfo != nullptr); | |||
for (uint32_t j=0, paramCount = carla_get_parameter_count(i); j < paramCount; ++j) | |||
{ | |||
const CarlaParameterInfo* const paramInfo(carla_get_parameter_info(i, j)); | |||
CARLA_SAFE_ASSERT_CONTINUE(paramInfo != nullptr); | |||
const ParameterData* const paramData(carla_get_parameter_data(i, j)); | |||
CARLA_SAFE_ASSERT_CONTINUE(paramData != nullptr); | |||
const ParameterRanges* const paramRanges(carla_get_parameter_ranges(i, j)); | |||
CARLA_SAFE_ASSERT_CONTINUE(paramRanges != nullptr); | |||
if (paramData->type != CB::PARAMETER_INPUT && paramData->type != CB::PARAMETER_OUTPUT) | |||
continue; | |||
if ((paramData->hints & CB::PARAMETER_IS_ENABLED) == 0) | |||
continue; | |||
if ((paramData->hints & CB::PARAMETER_IS_AUTOMABLE) == 0) | |||
continue; | |||
if (paramData->hints & CB::PARAMETER_IS_READ_ONLY) | |||
continue; | |||
if (lo_message msg2 = lo_message_new()) | |||
{ | |||
stripName = fClientNameId + "/strip/Unnamed/" + pluginInfo->name + "/" + paramInfo->name; | |||
stripName.replace(' ','_'); | |||
lo_message_add(msg2, "sssfff", | |||
path, | |||
stripName.buffer(), | |||
paramData->type == CB::PARAMETER_INPUT ? "in" : "out", | |||
paramRanges->min, | |||
paramRanges->max, | |||
paramRanges->def); | |||
lo_send_message_from(targetAddress, fOscServer, "/reply", msg2); | |||
lo_message_free(msg2); | |||
carla_stdout("CarlaNSM::handleBroadcast - sent list at %i : %i, path: %s", i+1, j+1, stripName.buffer()); | |||
} | |||
#if 0 | |||
if (prefix == nullptr /*|| strncmp(o->path(), prefix, std::strlen(prefix)) == 0*/) | |||
{ | |||
lo_send_from(targetAddress, fOscServer, LO_TT_IMMEDIATE, "/reply", "sssfff", | |||
path, | |||
paramInfo->name, | |||
paramData->type == CB::PARAMETER_INPUT ? "in" : "out", | |||
paramRanges->min, | |||
paramRanges->max, | |||
paramRanges->def); | |||
} | |||
#endif | |||
} | |||
} | |||
if (lo_message msg2 = lo_message_new()) | |||
{ | |||
lo_message_add(msg2, "s", path); | |||
lo_send_message_from(targetAddress, fOscServer, "/reply", msg2); | |||
lo_message_free(msg2); | |||
carla_stdout("CarlaNSM::handleBroadcast - sent list final"); | |||
} | |||
//lo_send_from(targetAddress, fOscServer, LO_TT_IMMEDIATE, "/reply", "s", path); | |||
//return 0; | |||
} | |||
#if 0 | |||
if (std::strcmp(path, "/reply") == 0) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(argc > 0, 0); | |||
const char* const method = &argv[0]->s; | |||
if (std::strcmp(method, "/signal/list") == 0) | |||
{ | |||
carla_stdout("CarlaNSM::handleBroadcast - got new list"); | |||
if (argc == 1) | |||
return 0; | |||
CARLA_SAFE_ASSERT_RETURN(argc == 6, 0); | |||
const lo_address targetAddress(lo_message_get_source(msg)); | |||
CARLA_SAFE_ASSERT_RETURN(targetAddress != nullptr, 0); | |||
const char* const stripName = &argv[1]->s; | |||
const char* const orientation = &argv[2]->s; | |||
const float min = argv[3]->f; | |||
const float max = argv[4]->f; | |||
const float def = argv[5]->f; | |||
if (lo_message msg2 = lo_message_new()) | |||
{ | |||
lo_message_add(msg2, "sssfff", | |||
method, | |||
stripName, | |||
orientation, | |||
min, | |||
max, | |||
def); | |||
lo_send_message_from(targetAddress, fOscServer, "/reply", msg2); | |||
lo_message_free(msg2); | |||
} | |||
} | |||
else | |||
{ | |||
CARLA_SAFE_ASSERT(false); | |||
} | |||
return 0; | |||
} | |||
#endif | |||
// check if all args are strings | |||
for (int i=0; i<argc; ++i) | |||
{ | |||
if (types[i] != 's') | |||
return 0; | |||
} | |||
for (int i=0; i<argc; ++i) | |||
carla_stdout("%i: %s", i+1, &argv[i]->s); | |||
return 0; | |||
} | |||
private: | |||
lo_address fOscAddress; | |||
lo_server fOscServer; | |||
lo_server_thread fOscServerThread; | |||
CarlaString fClientNameId; | |||
CarlaString fProjectPath; | |||
bool fHasBroadcast; | |||
bool fHasOptionalGui; | |||
bool fHasServerControl; | |||
bool fStarted; | |||
volatile bool fReadyActionOpen; | |||
volatile bool fReadyActionSave; | |||
#define handlePtr ((CarlaNSM*)data) | |||
static void _osc_error_handler(int num, const char* msg, const char* path) | |||
{ | |||
carla_stderr2("CarlaNSM::_osc_error_handler(%i, \"%s\", \"%s\")", num, msg, path); | |||
} | |||
static int _error_handler(const char*, const char* types, lo_arg** argv, int argc, lo_message, void* data) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(argc == 3, 1); | |||
CARLA_SAFE_ASSERT_RETURN(std::strcmp(types, "sis") == 0, 1); | |||
const char* const method = &argv[0]->s; | |||
const int code = argv[1]->i; | |||
const char* const message = &argv[2]->s; | |||
return handlePtr->handleError(method, code, message); | |||
} | |||
static int _reply_handler(const char*, const char* types, lo_arg** argv, int argc, lo_message msg, void* data) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(argc == 4, 1); | |||
CARLA_SAFE_ASSERT_RETURN(std::strcmp(types, "ssss") == 0, 1); | |||
const char* const method = &argv[0]->s; | |||
const char* const message = &argv[1]->s; | |||
const char* const smName = &argv[2]->s; | |||
const char* const features = &argv[3]->s; | |||
return handlePtr->handleReply(method, message, smName, features, msg); | |||
} | |||
static int _open_handler(const char*, const char* types, lo_arg** argv, int argc, lo_message, void* data) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(argc == 3, 1); | |||
CARLA_SAFE_ASSERT_RETURN(std::strcmp(types, "sss") == 0, 1); | |||
const char* const projectPath = &argv[0]->s; | |||
const char* const displayName = &argv[1]->s; | |||
const char* const clientNameId = &argv[2]->s; | |||
return handlePtr->handleOpen(projectPath, displayName, clientNameId); | |||
} | |||
static int _save_handler(const char*, const char*, lo_arg**, int argc, lo_message, void* data) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(argc == 0, 1); | |||
return handlePtr->handleSave(); | |||
} | |||
static int _loaded_handler(const char*, const char*, lo_arg**, int argc, lo_message, void* data) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(argc == 0, 1); | |||
return handlePtr->handleSessionIsLoaded(); | |||
} | |||
static int _show_gui_handler(const char*, const char*, lo_arg**, int argc, lo_message, void* data) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(argc == 0, 1); | |||
return handlePtr->handleShowOptionalGui(); | |||
} | |||
static int _hide_gui_handler(const char*, const char*, lo_arg**, int argc, lo_message, void* data) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(argc == 0, 1); | |||
return handlePtr->handleHideOptionalGui(); | |||
} | |||
static int _broadcast_handler(const char* path, const char* types, lo_arg** argv, int argc, lo_message msg, void* data) | |||
{ | |||
return handlePtr->handleBroadcast(path, types, argv, argc, msg); | |||
} | |||
#undef handlePtr | |||
CARLA_PREVENT_HEAP_ALLOCATION | |||
CARLA_DECLARE_NON_COPY_CLASS(CarlaNSM) | |||
}; | |||
#endif // HAVE_LIBLO | |||
CarlaBackendStandalone gStandalone; | |||
// ------------------------------------------------------------------------------------------------------------------- | |||
// API | |||
@@ -2564,27 +1914,6 @@ const char* carla_get_host_osc_url_udp() | |||
// ------------------------------------------------------------------------------------------------------------------- | |||
bool carla_nsm_init(int pid, const char* executableName) | |||
{ | |||
#ifdef HAVE_LIBLO | |||
return CarlaNSM::getInstance().announce(pid, executableName); | |||
#else | |||
return false; | |||
// unused | |||
(void)pid; (void)executableName; | |||
#endif | |||
} | |||
void carla_nsm_ready(int action) | |||
{ | |||
#ifdef HAVE_LIBLO | |||
CarlaNSM::getInstance().ready(action); | |||
#endif | |||
} | |||
// ------------------------------------------------------------------------------------------------------------------- | |||
#include "CarlaPluginUI.cpp" | |||
#include "CarlaDssiUtils.cpp" | |||
#include "CarlaPatchbayUtils.cpp" | |||
@@ -0,0 +1,712 @@ | |||
/* | |||
* Carla Standalone | |||
* Copyright (C) 2011-2015 Filipe Coelho <falktx@falktx.com> | |||
* | |||
* 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 "CarlaDefines.h" | |||
#ifdef HAVE_LIBLO | |||
#define NSM_API_VERSION_MAJOR 1 | |||
#define NSM_API_VERSION_MINOR 2 | |||
#define NSM_CLIENT_FEATURES ":switch:" | |||
//#define NSM_CLIENT_FEATURES ":switch:optional-gui:" | |||
#include "CarlaHost.h" | |||
#include "CarlaOscUtils.hpp" | |||
#include "CarlaString.hpp" | |||
#include "juce_core.h" | |||
namespace CB = CarlaBackend; | |||
// ------------------------------------------------------------------------------------------------------------------- | |||
struct CarlaBackendStandalone { | |||
CarlaEngine* engine; | |||
EngineCallbackFunc engineCallback; | |||
void* engineCallbackPtr; | |||
// ... | |||
}; | |||
extern CarlaBackendStandalone gStandalone; | |||
// ------------------------------------------------------------------------------------------------------------------- | |||
class CarlaNSM | |||
{ | |||
public: | |||
CarlaNSM() noexcept | |||
: fOscAddress(nullptr), | |||
fOscServer(nullptr), | |||
fOscServerThread(nullptr), | |||
fClientNameId(), | |||
fProjectPath(), | |||
fHasBroadcast(false), | |||
fHasOptionalGui(false), | |||
fHasServerControl(false), | |||
fStarted(), | |||
fReadyActionOpen(true), | |||
fReadyActionSave(true) {} | |||
~CarlaNSM() | |||
{ | |||
CARLA_SAFE_ASSERT(fReadyActionOpen); | |||
CARLA_SAFE_ASSERT(fReadyActionSave); | |||
if (fOscServerThread != nullptr) | |||
{ | |||
lo_server_thread_stop(fOscServerThread); | |||
lo_server_thread_free(fOscServerThread); | |||
fOscServerThread = nullptr; | |||
fOscServer = nullptr; | |||
} | |||
if (fOscAddress != nullptr) | |||
{ | |||
lo_address_free(fOscAddress); | |||
fOscAddress = nullptr; | |||
} | |||
} | |||
bool announce(const int pid, const char* const executableName) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(pid != 0, false); | |||
CARLA_SAFE_ASSERT_RETURN(executableName != nullptr && executableName[0] != '\0', false); | |||
const char* const NSM_URL(std::getenv("NSM_URL")); | |||
if (NSM_URL == nullptr) | |||
return false; | |||
const lo_address addr = lo_address_new_from_url(NSM_URL); | |||
CARLA_SAFE_ASSERT_RETURN(addr != nullptr, false); | |||
const int proto = lo_address_get_protocol(addr); | |||
if (fOscServerThread == nullptr) | |||
{ | |||
// create new OSC server | |||
fOscServerThread = lo_server_thread_new_with_proto(nullptr, proto, _osc_error_handler); | |||
CARLA_SAFE_ASSERT_RETURN(fOscServerThread != nullptr, false); | |||
// register message handlers | |||
lo_server_thread_add_method(fOscServerThread, "/error", "sis", _error_handler, this); | |||
lo_server_thread_add_method(fOscServerThread, "/reply", "ssss", _reply_handler, this); | |||
lo_server_thread_add_method(fOscServerThread, "/nsm/client/open", "sss", _open_handler, this); | |||
lo_server_thread_add_method(fOscServerThread, "/nsm/client/save", "", _save_handler, this); | |||
lo_server_thread_add_method(fOscServerThread, "/nsm/client/session_is_loaded", "", _loaded_handler, this); | |||
lo_server_thread_add_method(fOscServerThread, "/nsm/client/show_optional_gui", "", _show_gui_handler, this); | |||
lo_server_thread_add_method(fOscServerThread, "/nsm/client/hide_optional_gui", "", _hide_gui_handler, this); | |||
lo_server_thread_add_method(fOscServerThread, nullptr, nullptr, _broadcast_handler, this); | |||
fOscServer = lo_server_thread_get_server(fOscServerThread); | |||
} | |||
lo_send_from(addr, fOscServer, LO_TT_IMMEDIATE, "/nsm/server/announce", "sssiii", | |||
"Carla", NSM_CLIENT_FEATURES, executableName, NSM_API_VERSION_MAJOR, NSM_API_VERSION_MINOR, pid); | |||
lo_address_free(addr); | |||
return true; | |||
} | |||
void ready(const int action) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(fOscServerThread != nullptr,); | |||
switch (action) | |||
{ | |||
case -1: // init | |||
CARLA_SAFE_ASSERT_BREAK(! fStarted); | |||
fStarted = true; | |||
lo_server_thread_start(fOscServerThread); | |||
break; | |||
case 0: // error | |||
break; | |||
case 1: // reply | |||
break; | |||
case 2: // open | |||
fReadyActionOpen = true; | |||
break; | |||
case 3: // save | |||
fReadyActionSave = true; | |||
break; | |||
case 4: // session loaded | |||
break; | |||
case 5: // show gui | |||
CARLA_SAFE_ASSERT_BREAK(fOscAddress != nullptr); | |||
CARLA_SAFE_ASSERT_BREAK(fOscServer != nullptr); | |||
lo_send_from(fOscAddress, fOscServer, LO_TT_IMMEDIATE, "/nsm/client/gui_is_shown", ""); | |||
break; | |||
case 6: // hide gui | |||
CARLA_SAFE_ASSERT_BREAK(fOscAddress != nullptr); | |||
CARLA_SAFE_ASSERT_BREAK(fOscServer != nullptr); | |||
lo_send_from(fOscAddress, fOscServer, LO_TT_IMMEDIATE, "/nsm/client/gui_is_hidden", ""); | |||
break; | |||
} | |||
} | |||
static CarlaNSM& getInstance() | |||
{ | |||
static CarlaNSM sInstance; | |||
return sInstance; | |||
} | |||
protected: | |||
int handleError(const char* const method, const int code, const char* const message) | |||
{ | |||
carla_stdout("CarlaNSM::handleError(\"%s\", %i, \"%s\")", method, code, message); | |||
if (gStandalone.engineCallback != nullptr) | |||
gStandalone.engineCallback(gStandalone.engineCallbackPtr, CB::ENGINE_CALLBACK_NSM, 0, 0, code, 0.0f, message); | |||
return 1; | |||
// may be unused | |||
(void)method; | |||
} | |||
int handleReply(const char* const method, const char* const message, const char* const smName, const char* const features, | |||
const lo_message msg) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(fOscServerThread != nullptr, 1); | |||
carla_stdout("CarlaNSM::handleReply(\"%s\", \"%s\", \"%s\", \"%s\")", method, message, smName, features); | |||
if (std::strcmp(method, "/nsm/server/announce") == 0) | |||
{ | |||
char* const addressURL(lo_address_get_url(lo_message_get_source(msg))); | |||
CARLA_SAFE_ASSERT_RETURN(addressURL != nullptr, 0); | |||
if (fOscAddress != nullptr) | |||
lo_address_free(fOscAddress); | |||
fOscAddress = lo_address_new_from_url(addressURL); | |||
CARLA_SAFE_ASSERT_RETURN(fOscAddress != nullptr, 0); | |||
fHasBroadcast = std::strstr(features, ":broadcast:") != nullptr; | |||
fHasOptionalGui = std::strstr(features, ":optional-gui:") != nullptr; | |||
fHasServerControl = std::strstr(features, ":server_control:") != nullptr; | |||
// UI starts visible | |||
if (fHasOptionalGui) | |||
lo_send_from(fOscAddress, fOscServer, LO_TT_IMMEDIATE, "/nsm/client/gui_is_shown", ""); | |||
carla_stdout("Carla started via '%s', message: %s", smName, message); | |||
if (gStandalone.engineCallback != nullptr) | |||
{ | |||
int flags = 0; | |||
if (fHasBroadcast) | |||
flags |= 1 << 0; | |||
if (fHasOptionalGui) | |||
flags |= 1 << 1; | |||
if (fHasServerControl) | |||
flags |= 1 << 2; | |||
gStandalone.engineCallback(gStandalone.engineCallbackPtr, CB::ENGINE_CALLBACK_NSM, 0, 1, flags, 0.0f, smName); | |||
} | |||
} | |||
else | |||
{ | |||
carla_stdout("Got unknown NSM reply method '%s'", method); | |||
} | |||
return 0; | |||
} | |||
int handleOpen(const char* const projectPath, const char* const displayName, const char* const clientNameId) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(fOscAddress != nullptr, 1); | |||
CARLA_SAFE_ASSERT_RETURN(fOscServer != nullptr, 1); | |||
carla_stdout("CarlaNSM::handleOpen(\"%s\", \"%s\", \"%s\")", projectPath, displayName, clientNameId); | |||
if (gStandalone.engineCallback != nullptr) | |||
{ | |||
fReadyActionOpen = false; | |||
gStandalone.engineCallback(gStandalone.engineCallbackPtr, CB::ENGINE_CALLBACK_NSM, 0, 2, 0, 0.0f, projectPath); | |||
for (; ! fReadyActionOpen;) | |||
carla_msleep(10); | |||
} | |||
else | |||
{ | |||
using namespace juce; | |||
if (carla_is_engine_running()) | |||
carla_engine_close(); | |||
carla_engine_init("JACK", clientNameId); | |||
fProjectPath = projectPath; | |||
fProjectPath += ".carxp"; | |||
const String jfilename = String(CharPointer_UTF8(fProjectPath)); | |||
if (File(jfilename).existsAsFile()) | |||
carla_load_project(fProjectPath); | |||
} | |||
fClientNameId = clientNameId; | |||
lo_send_from(fOscAddress, fOscServer, LO_TT_IMMEDIATE, "/reply", "ss", "/nsm/client/open", "OK"); | |||
// Broadcast ourselves | |||
if (fHasBroadcast) | |||
{ | |||
char* const oscServerAddress(lo_server_get_url(fOscServer)); | |||
if (lo_message msg = lo_message_new()) | |||
{ | |||
lo_message_add(msg, "sssss", | |||
"/non/hello", | |||
oscServerAddress, | |||
"Carla", | |||
CARLA_VERSION_STRING, | |||
fClientNameId.buffer()); | |||
lo_send_message_from(fOscAddress, fOscServer, "/nsm/server/broadcast", msg); | |||
lo_message_free(msg); | |||
} | |||
std::free(oscServerAddress); | |||
} | |||
return 0; | |||
} | |||
int handleSave() | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(fOscAddress != nullptr, 1); | |||
CARLA_SAFE_ASSERT_RETURN(fOscServer != nullptr, 1); | |||
carla_stdout("CarlaNSM::handleSave()"); | |||
if (gStandalone.engineCallback != nullptr) | |||
{ | |||
fReadyActionSave = false; | |||
gStandalone.engineCallback(gStandalone.engineCallbackPtr, CB::ENGINE_CALLBACK_NSM, 0, 3, 0, 0.0f, nullptr); | |||
for (; ! fReadyActionSave;) | |||
carla_msleep(10); | |||
} | |||
else | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(fProjectPath.isNotEmpty(), 0); | |||
carla_save_project(fProjectPath); | |||
} | |||
lo_send_from(fOscAddress, fOscServer, LO_TT_IMMEDIATE, "/reply", "ss", "/nsm/client/save", "OK"); | |||
return 0; | |||
} | |||
int handleSessionIsLoaded() | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(fOscAddress != nullptr, 1); | |||
CARLA_SAFE_ASSERT_RETURN(fOscServer != nullptr, 1); | |||
carla_stdout("CarlaNSM::handleSessionIsLoaded()"); | |||
if (gStandalone.engineCallback != nullptr) | |||
gStandalone.engineCallback(gStandalone.engineCallbackPtr, CB::ENGINE_CALLBACK_NSM, 0, 4, 0, 0.0f, nullptr); | |||
return 0; | |||
} | |||
int handleShowOptionalGui() | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(fOscAddress != nullptr, 1); | |||
CARLA_SAFE_ASSERT_RETURN(fOscServer != nullptr, 1); | |||
carla_stdout("CarlaNSM::handleShowOptionalGui()"); | |||
if (gStandalone.engineCallback != nullptr) | |||
gStandalone.engineCallback(gStandalone.engineCallbackPtr, CB::ENGINE_CALLBACK_NSM, 0, 5, 0, 0.0f, nullptr); | |||
return 0; | |||
} | |||
int handleHideOptionalGui() | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(fOscAddress != nullptr, 1); | |||
CARLA_SAFE_ASSERT_RETURN(fOscServer != nullptr, 1); | |||
carla_stdout("CarlaNSM::handleHideOptionalGui()"); | |||
if (gStandalone.engineCallback != nullptr) | |||
gStandalone.engineCallback(gStandalone.engineCallbackPtr, CB::ENGINE_CALLBACK_NSM, 0, 6, 0, 0.0f, nullptr); | |||
return 0; | |||
} | |||
int handleBroadcast(const char* const path, const char* const types, lo_arg** const argv, const int argc, | |||
const lo_message msg) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(fOscAddress != nullptr, 1); | |||
CARLA_SAFE_ASSERT_RETURN(fOscServer != nullptr, 1); | |||
CARLA_SAFE_ASSERT_RETURN(argc >= 0, 0); | |||
carla_stdout("CarlaNSM::handleBroadcast(%s, %s, %p, %i)", path, types, argv, argc); | |||
if (std::strcmp(path, "/non/hello") == 0) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(argc == 4, 0); | |||
CARLA_SAFE_ASSERT_RETURN(std::strcmp(types, "ssss") == 0, 0); | |||
const char* const url = &argv[0]->s; | |||
//const char* const name = &argv[1]->s; | |||
//const char* const version = &argv[2]->s; | |||
//const char* const clientId = &argv[3]->s; | |||
const lo_address targetAddress(lo_address_new_from_url(url)); | |||
CARLA_SAFE_ASSERT_RETURN(targetAddress != nullptr, 0); | |||
char* const oscServerAddress(lo_server_get_url(fOscServer)); | |||
CARLA_SAFE_ASSERT_RETURN(oscServerAddress != nullptr, 0); | |||
if (lo_message msg2 = lo_message_new()) | |||
{ | |||
lo_message_add(msg2, "ss", | |||
fClientNameId.buffer(), | |||
oscServerAddress); | |||
lo_send_message_from(targetAddress, fOscServer, "/signal/hello", msg2); | |||
lo_message_free(msg2); | |||
carla_stdout("CarlaNSM::handleBroadcast - sent hello"); | |||
} | |||
lo_address_free(targetAddress); | |||
std::free(oscServerAddress); | |||
return 0; | |||
} | |||
if (std::strcmp(path, "/signal/hello") == 0) | |||
{ | |||
carla_stdout("CarlaNSM::handleBroadcast - got hello"); | |||
//const char* const name = &argv[0]->s; | |||
const char* const url = &argv[1]->s; | |||
const lo_address targetAddress(lo_address_new_from_url(url)); | |||
CARLA_SAFE_ASSERT_RETURN(targetAddress != nullptr, 0); | |||
char* const oscServerAddress(lo_server_get_url(fOscServer)); | |||
CARLA_SAFE_ASSERT_RETURN(oscServerAddress != nullptr, 0); | |||
//lo_send_from(targetAddress, fOscServer, LO_TT_IMMEDIATE, "/signal/list", ""); | |||
if (lo_message msg2 = lo_message_new()) | |||
{ | |||
lo_message_add(msg2, "ss", | |||
fClientNameId.buffer(), | |||
oscServerAddress); | |||
lo_send_message_from(targetAddress, fOscServer, "/signal/hello", msg2); | |||
lo_message_free(msg2); | |||
carla_stdout("CarlaNSM::handleBroadcast - sent hello"); | |||
} | |||
lo_address_free(targetAddress); | |||
std::free(oscServerAddress); | |||
return 0; | |||
} | |||
if (std::strcmp(path, "/signal/list") == 0) | |||
{ | |||
carla_stdout("CarlaNSM::handleBroadcast - got list"); | |||
CARLA_SAFE_ASSERT_RETURN(carla_is_engine_running(), 0); | |||
//const char* prefix = nullptr; | |||
//if (argc > 0) | |||
//prefix = &argv[0]->s; | |||
const lo_address targetAddress(lo_message_get_source(msg)); | |||
CARLA_SAFE_ASSERT_RETURN(targetAddress != nullptr, 0); | |||
CarlaString stripName; | |||
for (uint32_t i = 0, pluginCount = carla_get_current_plugin_count(); i < pluginCount; ++i) | |||
{ | |||
const CarlaPluginInfo* const pluginInfo(carla_get_plugin_info(i)); | |||
CARLA_SAFE_ASSERT_CONTINUE(pluginInfo != nullptr); | |||
for (uint32_t j=0, paramCount = carla_get_parameter_count(i); j < paramCount; ++j) | |||
{ | |||
const CarlaParameterInfo* const paramInfo(carla_get_parameter_info(i, j)); | |||
CARLA_SAFE_ASSERT_CONTINUE(paramInfo != nullptr); | |||
const ParameterData* const paramData(carla_get_parameter_data(i, j)); | |||
CARLA_SAFE_ASSERT_CONTINUE(paramData != nullptr); | |||
const ParameterRanges* const paramRanges(carla_get_parameter_ranges(i, j)); | |||
CARLA_SAFE_ASSERT_CONTINUE(paramRanges != nullptr); | |||
if (paramData->type != CB::PARAMETER_INPUT && paramData->type != CB::PARAMETER_OUTPUT) | |||
continue; | |||
if ((paramData->hints & CB::PARAMETER_IS_ENABLED) == 0) | |||
continue; | |||
if ((paramData->hints & CB::PARAMETER_IS_AUTOMABLE) == 0) | |||
continue; | |||
if (paramData->hints & CB::PARAMETER_IS_READ_ONLY) | |||
continue; | |||
if (lo_message msg2 = lo_message_new()) | |||
{ | |||
stripName = fClientNameId + "/strip/Unnamed/" + pluginInfo->name + "/" + paramInfo->name; | |||
stripName.replace(' ','_'); | |||
lo_message_add(msg2, "sssfff", | |||
path, | |||
stripName.buffer(), | |||
paramData->type == CB::PARAMETER_INPUT ? "in" : "out", | |||
paramRanges->min, | |||
paramRanges->max, | |||
paramRanges->def); | |||
lo_send_message_from(targetAddress, fOscServer, "/reply", msg2); | |||
lo_message_free(msg2); | |||
carla_stdout("CarlaNSM::handleBroadcast - sent list at %i : %i, path: %s", i+1, j+1, stripName.buffer()); | |||
} | |||
#if 0 | |||
if (prefix == nullptr /*|| strncmp(o->path(), prefix, std::strlen(prefix)) == 0*/) | |||
{ | |||
lo_send_from(targetAddress, fOscServer, LO_TT_IMMEDIATE, "/reply", "sssfff", | |||
path, | |||
paramInfo->name, | |||
paramData->type == CB::PARAMETER_INPUT ? "in" : "out", | |||
paramRanges->min, | |||
paramRanges->max, | |||
paramRanges->def); | |||
} | |||
#endif | |||
} | |||
} | |||
if (lo_message msg2 = lo_message_new()) | |||
{ | |||
lo_message_add(msg2, "s", path); | |||
lo_send_message_from(targetAddress, fOscServer, "/reply", msg2); | |||
lo_message_free(msg2); | |||
carla_stdout("CarlaNSM::handleBroadcast - sent list final"); | |||
} | |||
//lo_send_from(targetAddress, fOscServer, LO_TT_IMMEDIATE, "/reply", "s", path); | |||
//return 0; | |||
} | |||
#if 0 | |||
if (std::strcmp(path, "/reply") == 0) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(argc > 0, 0); | |||
const char* const method = &argv[0]->s; | |||
if (std::strcmp(method, "/signal/list") == 0) | |||
{ | |||
carla_stdout("CarlaNSM::handleBroadcast - got new list"); | |||
if (argc == 1) | |||
return 0; | |||
CARLA_SAFE_ASSERT_RETURN(argc == 6, 0); | |||
const lo_address targetAddress(lo_message_get_source(msg)); | |||
CARLA_SAFE_ASSERT_RETURN(targetAddress != nullptr, 0); | |||
const char* const stripName = &argv[1]->s; | |||
const char* const orientation = &argv[2]->s; | |||
const float min = argv[3]->f; | |||
const float max = argv[4]->f; | |||
const float def = argv[5]->f; | |||
if (lo_message msg2 = lo_message_new()) | |||
{ | |||
lo_message_add(msg2, "sssfff", | |||
method, | |||
stripName, | |||
orientation, | |||
min, | |||
max, | |||
def); | |||
lo_send_message_from(targetAddress, fOscServer, "/reply", msg2); | |||
lo_message_free(msg2); | |||
} | |||
} | |||
else | |||
{ | |||
CARLA_SAFE_ASSERT(false); | |||
} | |||
return 0; | |||
} | |||
#endif | |||
// check if all args are strings | |||
for (int i=0; i<argc; ++i) | |||
{ | |||
if (types[i] != 's') | |||
return 0; | |||
} | |||
for (int i=0; i<argc; ++i) | |||
carla_stdout("%i: %s", i+1, &argv[i]->s); | |||
return 0; | |||
} | |||
private: | |||
lo_address fOscAddress; | |||
lo_server fOscServer; | |||
lo_server_thread fOscServerThread; | |||
CarlaString fClientNameId; | |||
CarlaString fProjectPath; | |||
bool fHasBroadcast; | |||
bool fHasOptionalGui; | |||
bool fHasServerControl; | |||
bool fStarted; | |||
volatile bool fReadyActionOpen; | |||
volatile bool fReadyActionSave; | |||
#define handlePtr ((CarlaNSM*)data) | |||
static void _osc_error_handler(int num, const char* msg, const char* path) | |||
{ | |||
carla_stderr2("CarlaNSM::_osc_error_handler(%i, \"%s\", \"%s\")", num, msg, path); | |||
} | |||
static int _error_handler(const char*, const char* types, lo_arg** argv, int argc, lo_message, void* data) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(argc == 3, 1); | |||
CARLA_SAFE_ASSERT_RETURN(std::strcmp(types, "sis") == 0, 1); | |||
const char* const method = &argv[0]->s; | |||
const int code = argv[1]->i; | |||
const char* const message = &argv[2]->s; | |||
return handlePtr->handleError(method, code, message); | |||
} | |||
static int _reply_handler(const char*, const char* types, lo_arg** argv, int argc, lo_message msg, void* data) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(argc == 4, 1); | |||
CARLA_SAFE_ASSERT_RETURN(std::strcmp(types, "ssss") == 0, 1); | |||
const char* const method = &argv[0]->s; | |||
const char* const message = &argv[1]->s; | |||
const char* const smName = &argv[2]->s; | |||
const char* const features = &argv[3]->s; | |||
return handlePtr->handleReply(method, message, smName, features, msg); | |||
} | |||
static int _open_handler(const char*, const char* types, lo_arg** argv, int argc, lo_message, void* data) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(argc == 3, 1); | |||
CARLA_SAFE_ASSERT_RETURN(std::strcmp(types, "sss") == 0, 1); | |||
const char* const projectPath = &argv[0]->s; | |||
const char* const displayName = &argv[1]->s; | |||
const char* const clientNameId = &argv[2]->s; | |||
return handlePtr->handleOpen(projectPath, displayName, clientNameId); | |||
} | |||
static int _save_handler(const char*, const char*, lo_arg**, int argc, lo_message, void* data) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(argc == 0, 1); | |||
return handlePtr->handleSave(); | |||
} | |||
static int _loaded_handler(const char*, const char*, lo_arg**, int argc, lo_message, void* data) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(argc == 0, 1); | |||
return handlePtr->handleSessionIsLoaded(); | |||
} | |||
static int _show_gui_handler(const char*, const char*, lo_arg**, int argc, lo_message, void* data) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(argc == 0, 1); | |||
return handlePtr->handleShowOptionalGui(); | |||
} | |||
static int _hide_gui_handler(const char*, const char*, lo_arg**, int argc, lo_message, void* data) | |||
{ | |||
CARLA_SAFE_ASSERT_RETURN(argc == 0, 1); | |||
return handlePtr->handleHideOptionalGui(); | |||
} | |||
static int _broadcast_handler(const char* path, const char* types, lo_arg** argv, int argc, lo_message msg, void* data) | |||
{ | |||
return handlePtr->handleBroadcast(path, types, argv, argc, msg); | |||
} | |||
#undef handlePtr | |||
CARLA_PREVENT_HEAP_ALLOCATION | |||
CARLA_DECLARE_NON_COPY_CLASS(CarlaNSM) | |||
}; | |||
#endif // HAVE_LIBLO | |||
// ------------------------------------------------------------------------------------------------------------------- | |||
CARLA_EXPORT | |||
bool carla_nsm_init(int pid, const char* executableName); | |||
bool carla_nsm_init(int pid, const char* executableName) | |||
{ | |||
#ifdef HAVE_LIBLO | |||
return CarlaNSM::getInstance().announce(pid, executableName); | |||
#else | |||
return false; | |||
// unused | |||
(void)pid; (void)executableName; | |||
#endif | |||
} | |||
CARLA_EXPORT | |||
void carla_nsm_ready(int action); | |||
void carla_nsm_ready(int action) | |||
{ | |||
#ifdef HAVE_LIBLO | |||
CarlaNSM::getInstance().ready(action); | |||
#endif | |||
} | |||
// ------------------------------------------------------------------------------------------------------------------- |
@@ -9,8 +9,11 @@ include Makefile.mk | |||
# ---------------------------------------------------------------------------------------------------------------------------- | |||
OBJS = \ | |||
OBJS_standalone = \ | |||
$(OBJDIR)/CarlaStandalone.cpp.o \ | |||
$(OBJDIR)/CarlaStandaloneNSM.cpp.o | |||
OBJS_utils = \ | |||
$(OBJDIR)/CarlaUtils.cpp.o | |||
TARGETS = \ | |||
@@ -123,7 +126,7 @@ all: $(TARGETS) | |||
# ---------------------------------------------------------------------------------------------------------------------------- | |||
clean: | |||
rm -f $(OBJS) $(TARGETS) | |||
rm -f $(OBJS_standalone) $(OBJS_utils) $(TARGETS) | |||
$(MAKE) clean -C engine | |||
$(MAKE) clean -C plugin | |||
@@ -135,15 +138,15 @@ doxygen: CarlaBackend.doxygen | |||
# ---------------------------------------------------------------------------------------------------------------------------- | |||
$(BINDIR)/libcarla_standalone2$(LIB_EXT): $(OBJDIR)/CarlaStandalone.cpp.o $(STANDALONE_LIBS) | |||
$(BINDIR)/libcarla_standalone2$(LIB_EXT): $(OBJS_standalone) $(STANDALONE_LIBS) | |||
-@mkdir -p $(BINDIR) | |||
@echo "Linking libcarla_standalone2$(LIB_EXT)" | |||
@$(CXX) $< $(LIBS_START) $(STANDALONE_LIBS) $(LIBS_END) $(LINK_FLAGS) $(STANDALONE_LINK_FLAGS) $(SHARED) -o $@ | |||
@$(CXX) $(OBJS_standalone) $(LIBS_START) $(STANDALONE_LIBS) $(LIBS_END) $(LINK_FLAGS) $(STANDALONE_LINK_FLAGS) $(SHARED) -o $@ | |||
$(BINDIR)/libcarla_utils$(LIB_EXT): $(OBJDIR)/CarlaUtils.cpp.o $(UTILS_LIBS) | |||
$(BINDIR)/libcarla_utils$(LIB_EXT): $(OBJS_utils) $(UTILS_LIBS) | |||
-@mkdir -p $(BINDIR) | |||
@echo "Linking libcarla_utils$(LIB_EXT)" | |||
@$(CXX) $< $(LIBS_START) $(UTILS_LIBS) $(LIBS_END) $(LINK_FLAGS) $(UTILS_LINK_FLAGS) $(SHARED) -o $@ | |||
@$(CXX) $(OBJS_utils) $(LIBS_START) $(UTILS_LIBS) $(LIBS_END) $(LINK_FLAGS) $(UTILS_LINK_FLAGS) $(SHARED) -o $@ | |||
# ---------------------------------------------------------------------------------------------------------------------------- | |||