diff DPF-Prymula-audioplugins/dpf/distrho/src/jackbridge/JackBridge.cpp @ 3:84e66ea83026

DPF-Prymula-audioplugins-0.231015-2
author prymula <prymula76@outlook.com>
date Mon, 16 Oct 2023 21:53:34 +0200
parents
children
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/DPF-Prymula-audioplugins/dpf/distrho/src/jackbridge/JackBridge.cpp	Mon Oct 16 21:53:34 2023 +0200
@@ -0,0 +1,2463 @@
+/*
+ * JackBridge for DPF
+ * Copyright (C) 2013-2022 Filipe Coelho <falktx@falktx.com>
+ *
+ * Permission to use, copy, modify, and/or distribute this software for any purpose with
+ * or without fee is hereby granted, provided that the above copyright notice and this
+ * permission notice appear in all copies.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD
+ * TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN
+ * NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
+ * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER
+ * IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
+ * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+ */
+
+#include "JackBridge.hpp"
+#include "../../DistrhoStandaloneUtils.hpp"
+
+#if ! (defined(JACKBRIDGE_DIRECT) || defined(JACKBRIDGE_DUMMY))
+
+#ifdef __WINE__
+# ifndef NOMINMAX
+#  define NOMINMAX
+# endif
+# include <windows.h>
+# include <pthread.h>
+#endif
+
+#if defined(DISTRHO_OS_WINDOWS) && ! defined(__WINE__)
+# define JACKSYM_API __cdecl
+#else
+# define JACKSYM_API
+#endif
+
+#include <cerrno>
+
+#ifdef HAVE_JACK
+# include "../../extra/LibraryUtils.hpp"
+#else
+typedef void* lib_t;
+#endif
+
+// in case JACK fails, we fallback to native bridges simulating JACK API
+#include "NativeBridge.hpp"
+
+#if defined(DISTRHO_OS_WASM)
+# include "WebBridge.hpp"
+#endif
+
+#ifndef DISTRHO_PROPER_CPP11_SUPPORT
+# undef HAVE_RTAUDIO
+#endif
+
+#ifdef DPF_JACK_STANDALONE_SKIP_RTAUDIO_FALLBACK
+# undef HAVE_RTAUDIO
+#endif
+
+#ifdef DPF_JACK_STANDALONE_SKIP_SDL2_FALLBACK
+# undef HAVE_SDL2
+#endif
+
+#if defined(HAVE_RTAUDIO) && DISTRHO_PLUGIN_NUM_INPUTS+DISTRHO_PLUGIN_NUM_OUTPUTS > 0
+// fix conflict between DGL and macOS names
+# define Point CorePoint
+# define Size CoreSize
+# include "RtAudioBridge.hpp"
+# ifdef RTAUDIO_API_TYPE
+#  include "rtaudio/RtAudio.cpp"
+# endif
+# ifdef RTMIDI_API_TYPE
+#  include "rtmidi/RtMidi.cpp"
+# endif
+# undef Point
+# undef Size
+#endif
+
+#if defined(HAVE_SDL2) && DISTRHO_PLUGIN_NUM_INPUTS+DISTRHO_PLUGIN_NUM_OUTPUTS > 0
+# include "SDL2Bridge.hpp"
+#endif
+
+// -----------------------------------------------------------------------------
+
+extern "C" {
+
+typedef void (JACKSYM_API *JackSymLatencyCallback)(jack_latency_callback_mode_t, void*);
+typedef int  (JACKSYM_API *JackSymProcessCallback)(jack_nframes_t, void*);
+typedef void (JACKSYM_API *JackSymThreadInitCallback)(void*);
+typedef int  (JACKSYM_API *JackSymGraphOrderCallback)(void*);
+typedef int  (JACKSYM_API *JackSymXRunCallback)(void*);
+typedef int  (JACKSYM_API *JackSymBufferSizeCallback)(jack_nframes_t, void*);
+typedef int  (JACKSYM_API *JackSymSampleRateCallback)(jack_nframes_t, void*);
+typedef void (JACKSYM_API *JackSymPortRegistrationCallback)(jack_port_id_t, int, void*);
+typedef void (JACKSYM_API *JackSymClientRegistrationCallback)(const char*, int, void*);
+typedef void (JACKSYM_API *JackSymPortConnectCallback)(jack_port_id_t, jack_port_id_t, int, void*);
+typedef void (JACKSYM_API *JackSymPortRenameCallback)(jack_port_id_t, const char*, const char*, void*);
+typedef void (JACKSYM_API *JackSymFreewheelCallback)(int, void*);
+typedef void (JACKSYM_API *JackSymShutdownCallback)(void*);
+typedef void (JACKSYM_API *JackSymInfoShutdownCallback)(jack_status_t, const char*, void*);
+typedef int  (JACKSYM_API *JackSymSyncCallback)(jack_transport_state_t, jack_position_t*, void*);
+typedef void (JACKSYM_API *JackSymTimebaseCallback)(jack_transport_state_t, jack_nframes_t, jack_position_t*, int, void*);
+typedef void (JACKSYM_API *JackSymSessionCallback)(jack_session_event_t*, void*);
+typedef void (JACKSYM_API *JackSymPropertyChangeCallback)(jack_uuid_t, const char*, jack_property_change_t, void*);
+typedef void* (JACKSYM_API *JackSymThreadCallback)(void*);
+
+typedef void        (JACKSYM_API *jacksym_get_version)(int*, int*, int*, int*);
+typedef const char* (JACKSYM_API *jacksym_get_version_string)(void);
+
+typedef jack_client_t* (JACKSYM_API *jacksym_client_open)(const char*, jack_options_t, jack_status_t*, ...);
+typedef int            (JACKSYM_API *jacksym_client_close)(jack_client_t*);
+
+typedef int   (JACKSYM_API *jacksym_client_name_size)(void);
+typedef char* (JACKSYM_API *jacksym_get_client_name)(jack_client_t*);
+
+typedef char* (JACKSYM_API *jacksym_client_get_uuid)(jack_client_t*);
+typedef char* (JACKSYM_API *jacksym_get_uuid_for_client_name)(jack_client_t*, const char*);
+typedef char* (JACKSYM_API *jacksym_get_client_name_by_uuid)(jack_client_t*, const char*);
+
+typedef int  (JACKBRIDGE_API *jacksym_uuid_parse)(const char*, jack_uuid_t*);
+typedef void (JACKBRIDGE_API *jacksym_uuid_unparse)(jack_uuid_t, char buf[JACK_UUID_STRING_SIZE]);
+
+typedef int (JACKSYM_API *jacksym_activate)(jack_client_t*);
+typedef int (JACKSYM_API *jacksym_deactivate)(jack_client_t*);
+typedef int (JACKSYM_API *jacksym_is_realtime)(jack_client_t*);
+
+typedef int  (JACKSYM_API *jacksym_set_thread_init_callback)(jack_client_t*, JackSymThreadInitCallback, void*);
+typedef void (JACKSYM_API *jacksym_on_shutdown)(jack_client_t*, JackSymShutdownCallback, void*);
+typedef void (JACKSYM_API *jacksym_on_info_shutdown)(jack_client_t*, JackSymInfoShutdownCallback, void*);
+typedef int  (JACKSYM_API *jacksym_set_process_callback)(jack_client_t*, JackSymProcessCallback, void*);
+typedef int  (JACKSYM_API *jacksym_set_freewheel_callback)(jack_client_t*, JackSymFreewheelCallback, void*);
+typedef int  (JACKSYM_API *jacksym_set_buffer_size_callback)(jack_client_t*, JackSymBufferSizeCallback, void*);
+typedef int  (JACKSYM_API *jacksym_set_sample_rate_callback)(jack_client_t*, JackSymSampleRateCallback, void*);
+typedef int  (JACKSYM_API *jacksym_set_client_registration_callback)(jack_client_t*, JackSymClientRegistrationCallback, void*);
+typedef int  (JACKSYM_API *jacksym_set_port_registration_callback)(jack_client_t*, JackSymPortRegistrationCallback, void*);
+typedef int  (JACKSYM_API *jacksym_set_port_rename_callback)(jack_client_t*, JackSymPortRenameCallback, void*);
+typedef int  (JACKSYM_API *jacksym_set_port_connect_callback)(jack_client_t*, JackSymPortConnectCallback, void*);
+typedef int  (JACKSYM_API *jacksym_set_graph_order_callback)(jack_client_t*, JackSymGraphOrderCallback, void*);
+typedef int  (JACKSYM_API *jacksym_set_xrun_callback)(jack_client_t*, JackSymXRunCallback, void*);
+typedef int  (JACKSYM_API *jacksym_set_latency_callback)(jack_client_t*, JackSymLatencyCallback, void*);
+
+typedef int (JACKSYM_API *jacksym_set_freewheel)(jack_client_t*, int);
+typedef int (JACKSYM_API *jacksym_set_buffer_size)(jack_client_t*, jack_nframes_t);
+
+typedef jack_nframes_t (JACKSYM_API *jacksym_get_sample_rate)(jack_client_t*);
+typedef jack_nframes_t (JACKSYM_API *jacksym_get_buffer_size)(jack_client_t*);
+typedef float          (JACKSYM_API *jacksym_cpu_load)(jack_client_t*);
+
+typedef jack_port_t* (JACKSYM_API *jacksym_port_register)(jack_client_t*, const char*, const char*, ulong, ulong);
+typedef int          (JACKSYM_API *jacksym_port_unregister)(jack_client_t*, jack_port_t*);
+typedef void*        (JACKSYM_API *jacksym_port_get_buffer)(jack_port_t*, jack_nframes_t);
+
+typedef const char*  (JACKSYM_API *jacksym_port_name)(const jack_port_t*);
+typedef jack_uuid_t  (JACKSYM_API *jacksym_port_uuid)(const jack_port_t*);
+typedef const char*  (JACKSYM_API *jacksym_port_short_name)(const jack_port_t*);
+typedef int          (JACKSYM_API *jacksym_port_flags)(const jack_port_t*);
+typedef const char*  (JACKSYM_API *jacksym_port_type)(const jack_port_t*);
+typedef int          (JACKSYM_API *jacksym_port_is_mine)(const jack_client_t*, const jack_port_t*);
+typedef int          (JACKSYM_API *jacksym_port_connected)(const jack_port_t*);
+typedef int          (JACKSYM_API *jacksym_port_connected_to)(const jack_port_t*, const char*);
+typedef const char** (JACKSYM_API *jacksym_port_get_connections)(const jack_port_t*);
+typedef const char** (JACKSYM_API *jacksym_port_get_all_connections)(const jack_client_t*, const jack_port_t*);
+
+typedef int (JACKSYM_API *jacksym_port_rename)(jack_client_t*, jack_port_t*, const char*);
+typedef int (JACKSYM_API *jacksym_port_set_name)(jack_port_t*, const char*);
+typedef int (JACKSYM_API *jacksym_port_set_alias)(jack_port_t*, const char*);
+typedef int (JACKSYM_API *jacksym_port_unset_alias)(jack_port_t*, const char*);
+typedef int (JACKSYM_API *jacksym_port_get_aliases)(const jack_port_t*, char* const aliases[2]);
+
+typedef int (JACKSYM_API *jacksym_port_request_monitor)(jack_port_t*, int);
+typedef int (JACKSYM_API *jacksym_port_request_monitor_by_name)(jack_client_t*, const char*, int);
+typedef int (JACKSYM_API *jacksym_port_ensure_monitor)(jack_port_t*, int);
+typedef int (JACKSYM_API *jacksym_port_monitoring_input)(jack_port_t*);
+
+typedef int (JACKSYM_API *jacksym_connect)(jack_client_t*, const char*, const char*);
+typedef int (JACKSYM_API *jacksym_disconnect)(jack_client_t*, const char*, const char*);
+typedef int (JACKSYM_API *jacksym_port_disconnect)(jack_client_t*, jack_port_t*);
+
+typedef int    (JACKSYM_API *jacksym_port_name_size)(void);
+typedef int    (JACKSYM_API *jacksym_port_type_size)(void);
+typedef size_t (JACKSYM_API *jacksym_port_type_get_buffer_size)(jack_client_t*, const char*);
+
+typedef void (JACKSYM_API *jacksym_port_get_latency_range)(jack_port_t*, jack_latency_callback_mode_t, jack_latency_range_t*);
+typedef void (JACKSYM_API *jacksym_port_set_latency_range)(jack_port_t*, jack_latency_callback_mode_t, jack_latency_range_t*);
+typedef int  (JACKSYM_API *jacksym_recompute_total_latencies)(jack_client_t*);
+
+typedef const char** (JACKSYM_API *jacksym_get_ports)(jack_client_t*, const char*, const char*, ulong);
+typedef jack_port_t* (JACKSYM_API *jacksym_port_by_name)(jack_client_t*, const char*);
+typedef jack_port_t* (JACKSYM_API *jacksym_port_by_id)(jack_client_t*, jack_port_id_t);
+
+typedef void (JACKSYM_API *jacksym_free)(void*);
+
+typedef uint32_t (JACKSYM_API *jacksym_midi_get_event_count)(void*);
+typedef int      (JACKSYM_API *jacksym_midi_event_get)(jack_midi_event_t*, void*, uint32_t);
+typedef void     (JACKSYM_API *jacksym_midi_clear_buffer)(void*);
+typedef int      (JACKSYM_API *jacksym_midi_event_write)(void*, jack_nframes_t, const jack_midi_data_t*, size_t);
+typedef jack_midi_data_t* (JACKSYM_API *jacksym_midi_event_reserve)(void*, jack_nframes_t, size_t);
+
+typedef int (JACKSYM_API *jacksym_release_timebase)(jack_client_t*);
+typedef int (JACKSYM_API *jacksym_set_sync_callback)(jack_client_t*, JackSymSyncCallback, void*);
+typedef int (JACKSYM_API *jacksym_set_sync_timeout)(jack_client_t*, jack_time_t);
+typedef int (JACKSYM_API *jacksym_set_timebase_callback)(jack_client_t*, int, JackSymTimebaseCallback, void*);
+typedef int (JACKSYM_API *jacksym_transport_locate)(jack_client_t*, jack_nframes_t);
+
+typedef jack_transport_state_t (JACKSYM_API *jacksym_transport_query)(const jack_client_t*, jack_position_t*);
+typedef jack_nframes_t         (JACKSYM_API *jacksym_get_current_transport_frame)(const jack_client_t*);
+
+typedef int  (JACKSYM_API *jacksym_transport_reposition)(jack_client_t*, const jack_position_t*);
+typedef void (JACKSYM_API *jacksym_transport_start)(jack_client_t*);
+typedef void (JACKSYM_API *jacksym_transport_stop)(jack_client_t*);
+
+typedef int  (JACKSYM_API *jacksym_set_property)(jack_client_t*, jack_uuid_t, const char*, const char*, const char*);
+typedef int  (JACKSYM_API *jacksym_get_property)(jack_uuid_t, const char*, char**, char**);
+typedef void (JACKSYM_API *jacksym_free_description)(jack_description_t*, int);
+typedef int  (JACKSYM_API *jacksym_get_properties)(jack_uuid_t, jack_description_t*);
+typedef int  (JACKSYM_API *jacksym_get_all_properties)(jack_description_t**);
+typedef int  (JACKSYM_API *jacksym_remove_property)(jack_client_t*, jack_uuid_t, const char*);
+typedef int  (JACKSYM_API *jacksym_remove_properties)(jack_client_t*, jack_uuid_t);
+typedef int  (JACKSYM_API *jacksym_remove_all_properties)(jack_client_t*);
+typedef int  (JACKSYM_API *jacksym_set_property_change_callback)(jack_client_t*, JackSymPropertyChangeCallback, void*);
+
+typedef bool           (JACKSYM_API *jacksym_set_process_thread)(jack_client_t*, JackSymThreadCallback callback, void*);
+typedef jack_nframes_t (JACKSYM_API *jacksym_cycle_wait)(jack_client_t*);
+typedef void           (JACKSYM_API *jacksym_cycle_signal)(jack_client_t*, int);
+
+#ifdef __WINE__
+typedef int  (JACKSYM_API *jacksym_thread_creator_t)(pthread_t*, const pthread_attr_t*, void *(*)(void*), void*);
+typedef void (JACKSYM_API *jacksym_set_thread_creator)(jacksym_thread_creator_t);
+#endif
+
+} // extern "C"
+
+// -----------------------------------------------------------------------------
+
+struct JackBridge {
+    lib_t lib;
+
+    jacksym_get_version get_version_ptr;
+    jacksym_get_version_string get_version_string_ptr;
+
+    jacksym_client_open client_open_ptr;
+    jacksym_client_close client_close_ptr;
+
+    jacksym_client_name_size client_name_size_ptr;
+    jacksym_get_client_name get_client_name_ptr;
+
+    jacksym_client_get_uuid client_get_uuid_ptr;
+    jacksym_get_uuid_for_client_name get_uuid_for_client_name_ptr;
+    jacksym_get_client_name_by_uuid get_client_name_by_uuid_ptr;
+
+    jacksym_uuid_parse uuid_parse_ptr;
+    jacksym_uuid_unparse uuid_unparse_ptr;
+
+    jacksym_activate activate_ptr;
+    jacksym_deactivate deactivate_ptr;
+    jacksym_is_realtime is_realtime_ptr;
+
+    jacksym_set_thread_init_callback set_thread_init_callback_ptr;
+    jacksym_on_shutdown on_shutdown_ptr;
+    jacksym_on_info_shutdown on_info_shutdown_ptr;
+    jacksym_set_process_callback set_process_callback_ptr;
+    jacksym_set_freewheel_callback set_freewheel_callback_ptr;
+    jacksym_set_buffer_size_callback set_buffer_size_callback_ptr;
+    jacksym_set_sample_rate_callback set_sample_rate_callback_ptr;
+    jacksym_set_client_registration_callback set_client_registration_callback_ptr;
+    jacksym_set_port_registration_callback set_port_registration_callback_ptr;
+    jacksym_set_port_rename_callback set_port_rename_callback_ptr;
+    jacksym_set_port_connect_callback set_port_connect_callback_ptr;
+    jacksym_set_graph_order_callback set_graph_order_callback_ptr;
+    jacksym_set_xrun_callback set_xrun_callback_ptr;
+    jacksym_set_latency_callback set_latency_callback_ptr;
+
+    jacksym_set_freewheel set_freewheel_ptr;
+    jacksym_set_buffer_size set_buffer_size_ptr;
+
+    jacksym_get_sample_rate get_sample_rate_ptr;
+    jacksym_get_buffer_size get_buffer_size_ptr;
+    jacksym_cpu_load cpu_load_ptr;
+
+    jacksym_port_register port_register_ptr;
+    jacksym_port_unregister port_unregister_ptr;
+    jacksym_port_get_buffer port_get_buffer_ptr;
+
+    jacksym_port_name port_name_ptr;
+    jacksym_port_uuid port_uuid_ptr;
+    jacksym_port_short_name port_short_name_ptr;
+    jacksym_port_flags port_flags_ptr;
+    jacksym_port_type port_type_ptr;
+    jacksym_port_is_mine port_is_mine_ptr;
+    jacksym_port_connected port_connected_ptr;
+    jacksym_port_connected_to port_connected_to_ptr;
+    jacksym_port_get_connections port_get_connections_ptr;
+    jacksym_port_get_all_connections port_get_all_connections_ptr;
+
+    jacksym_port_rename port_rename_ptr;
+    jacksym_port_set_name port_set_name_ptr;
+    jacksym_port_set_alias port_set_alias_ptr;
+    jacksym_port_unset_alias port_unset_alias_ptr;
+    jacksym_port_get_aliases port_get_aliases_ptr;
+
+    jacksym_port_request_monitor port_request_monitor_ptr;
+    jacksym_port_request_monitor_by_name port_request_monitor_by_name_ptr;
+    jacksym_port_ensure_monitor port_ensure_monitor_ptr;
+    jacksym_port_monitoring_input port_monitoring_input_ptr;
+
+    jacksym_connect connect_ptr;
+    jacksym_disconnect disconnect_ptr;
+    jacksym_port_disconnect port_disconnect_ptr;
+
+    jacksym_port_name_size port_name_size_ptr;
+    jacksym_port_type_size port_type_size_ptr;
+    jacksym_port_type_get_buffer_size port_type_get_buffer_size_ptr;
+
+    jacksym_port_get_latency_range port_get_latency_range_ptr;
+    jacksym_port_set_latency_range port_set_latency_range_ptr;
+    jacksym_recompute_total_latencies recompute_total_latencies_ptr;
+
+    jacksym_get_ports get_ports_ptr;
+    jacksym_port_by_name port_by_name_ptr;
+    jacksym_port_by_id port_by_id_ptr;
+
+    jacksym_free free_ptr;
+
+    jacksym_midi_get_event_count midi_get_event_count_ptr;
+    jacksym_midi_event_get midi_event_get_ptr;
+    jacksym_midi_clear_buffer midi_clear_buffer_ptr;
+    jacksym_midi_event_write midi_event_write_ptr;
+    jacksym_midi_event_reserve midi_event_reserve_ptr;
+
+    jacksym_release_timebase release_timebase_ptr;
+    jacksym_set_sync_callback set_sync_callback_ptr;
+    jacksym_set_sync_timeout set_sync_timeout_ptr;
+    jacksym_set_timebase_callback set_timebase_callback_ptr;
+    jacksym_transport_locate transport_locate_ptr;
+
+    jacksym_transport_query transport_query_ptr;
+    jacksym_get_current_transport_frame get_current_transport_frame_ptr;
+
+    jacksym_transport_reposition transport_reposition_ptr;
+    jacksym_transport_start transport_start_ptr;
+    jacksym_transport_stop transport_stop_ptr;
+
+    jacksym_set_property set_property_ptr;
+    jacksym_get_property get_property_ptr;
+    jacksym_free_description free_description_ptr;
+    jacksym_get_properties get_properties_ptr;
+    jacksym_get_all_properties get_all_properties_ptr;
+    jacksym_remove_property remove_property_ptr;
+    jacksym_remove_properties remove_properties_ptr;
+    jacksym_remove_all_properties remove_all_properties_ptr;
+    jacksym_set_property_change_callback set_property_change_callback_ptr;
+
+    jacksym_set_process_thread set_process_thread_ptr;
+    jacksym_cycle_wait cycle_wait_ptr;
+    jacksym_cycle_signal cycle_signal_ptr;
+
+   #ifdef __WINE__
+    jacksym_set_thread_creator set_thread_creator_ptr;
+   #endif
+
+    JackBridge()
+        : lib(nullptr),
+          get_version_ptr(nullptr),
+          get_version_string_ptr(nullptr),
+          client_open_ptr(nullptr),
+          client_close_ptr(nullptr),
+          client_name_size_ptr(nullptr),
+          get_client_name_ptr(nullptr),
+          client_get_uuid_ptr(nullptr),
+          get_uuid_for_client_name_ptr(nullptr),
+          get_client_name_by_uuid_ptr(nullptr),
+          uuid_parse_ptr(nullptr),
+          uuid_unparse_ptr(nullptr),
+          activate_ptr(nullptr),
+          deactivate_ptr(nullptr),
+          is_realtime_ptr(nullptr),
+          set_thread_init_callback_ptr(nullptr),
+          on_shutdown_ptr(nullptr),
+          on_info_shutdown_ptr(nullptr),
+          set_process_callback_ptr(nullptr),
+          set_freewheel_callback_ptr(nullptr),
+          set_buffer_size_callback_ptr(nullptr),
+          set_sample_rate_callback_ptr(nullptr),
+          set_client_registration_callback_ptr(nullptr),
+          set_port_registration_callback_ptr(nullptr),
+          set_port_rename_callback_ptr(nullptr),
+          set_port_connect_callback_ptr(nullptr),
+          set_graph_order_callback_ptr(nullptr),
+          set_xrun_callback_ptr(nullptr),
+          set_latency_callback_ptr(nullptr),
+          set_freewheel_ptr(nullptr),
+          set_buffer_size_ptr(nullptr),
+          get_sample_rate_ptr(nullptr),
+          get_buffer_size_ptr(nullptr),
+          cpu_load_ptr(nullptr),
+          port_register_ptr(nullptr),
+          port_unregister_ptr(nullptr),
+          port_get_buffer_ptr(nullptr),
+          port_name_ptr(nullptr),
+          port_uuid_ptr(nullptr),
+          port_short_name_ptr(nullptr),
+          port_flags_ptr(nullptr),
+          port_type_ptr(nullptr),
+          port_is_mine_ptr(nullptr),
+          port_connected_ptr(nullptr),
+          port_connected_to_ptr(nullptr),
+          port_get_connections_ptr(nullptr),
+          port_get_all_connections_ptr(nullptr),
+          port_rename_ptr(nullptr),
+          port_set_name_ptr(nullptr),
+          port_set_alias_ptr(nullptr),
+          port_unset_alias_ptr(nullptr),
+          port_get_aliases_ptr(nullptr),
+          port_request_monitor_ptr(nullptr),
+          port_request_monitor_by_name_ptr(nullptr),
+          port_ensure_monitor_ptr(nullptr),
+          port_monitoring_input_ptr(nullptr),
+          connect_ptr(nullptr),
+          disconnect_ptr(nullptr),
+          port_disconnect_ptr(nullptr),
+          port_name_size_ptr(nullptr),
+          port_type_size_ptr(nullptr),
+          port_type_get_buffer_size_ptr(nullptr),
+          port_get_latency_range_ptr(nullptr),
+          port_set_latency_range_ptr(nullptr),
+          recompute_total_latencies_ptr(nullptr),
+          get_ports_ptr(nullptr),
+          port_by_name_ptr(nullptr),
+          port_by_id_ptr(nullptr),
+          free_ptr(nullptr),
+          midi_get_event_count_ptr(nullptr),
+          midi_event_get_ptr(nullptr),
+          midi_clear_buffer_ptr(nullptr),
+          midi_event_write_ptr(nullptr),
+          midi_event_reserve_ptr(nullptr),
+          release_timebase_ptr(nullptr),
+          set_sync_callback_ptr(nullptr),
+          set_sync_timeout_ptr(nullptr),
+          set_timebase_callback_ptr(nullptr),
+          transport_locate_ptr(nullptr),
+          transport_query_ptr(nullptr),
+          get_current_transport_frame_ptr(nullptr),
+          transport_reposition_ptr(nullptr),
+          transport_start_ptr(nullptr),
+          transport_stop_ptr(nullptr),
+          set_property_ptr(nullptr),
+          get_property_ptr(nullptr),
+          free_description_ptr(nullptr),
+          get_properties_ptr(nullptr),
+          get_all_properties_ptr(nullptr),
+          remove_property_ptr(nullptr),
+          remove_properties_ptr(nullptr),
+          remove_all_properties_ptr(nullptr),
+          set_property_change_callback_ptr(nullptr),
+          set_process_thread_ptr(nullptr),
+          cycle_wait_ptr(nullptr),
+          cycle_signal_ptr(nullptr)
+         #ifdef __WINE__
+        , set_thread_creator_ptr(nullptr)
+         #endif
+    {
+      #ifdef HAVE_JACK
+       #if defined(DISTRHO_OS_MAC)
+        const char* const filename = "libjack.dylib";
+       #elif defined(DISTRHO_OS_WINDOWS) && defined(_WIN64)
+        const char* const filename = "libjack64.dll";
+       #elif defined(DISTRHO_OS_WINDOWS)
+        const char* const filename = "libjack.dll";
+       #else
+        const char* const filename = "libjack.so.0";
+       #endif
+
+        USE_NAMESPACE_DISTRHO
+
+        lib = lib_open(filename);
+
+        if (lib == nullptr)
+        {
+            fprintf(stderr, "Failed to load JACK DLL, reason:\n%s\n", lib_error(filename));
+            return;
+        }
+        else
+        {
+            fprintf(stdout, "%s loaded successfully!\n", filename);
+        }
+
+        #define JOIN(a, b) a ## b
+        #define LIB_SYMBOL(NAME) JOIN(NAME, _ptr) = lib_symbol<jacksym_##NAME>(lib, "jack_" #NAME);
+
+        LIB_SYMBOL(get_version)
+        LIB_SYMBOL(get_version_string)
+
+        LIB_SYMBOL(client_open)
+        LIB_SYMBOL(client_close)
+
+        LIB_SYMBOL(client_name_size)
+        LIB_SYMBOL(get_client_name)
+
+        LIB_SYMBOL(client_get_uuid)
+        LIB_SYMBOL(get_uuid_for_client_name)
+        LIB_SYMBOL(get_client_name_by_uuid)
+
+        LIB_SYMBOL(uuid_parse)
+        LIB_SYMBOL(uuid_unparse)
+
+        LIB_SYMBOL(activate)
+        LIB_SYMBOL(deactivate)
+        LIB_SYMBOL(is_realtime)
+
+        LIB_SYMBOL(set_thread_init_callback)
+        LIB_SYMBOL(on_shutdown)
+        LIB_SYMBOL(on_info_shutdown)
+        LIB_SYMBOL(set_process_callback)
+        LIB_SYMBOL(set_freewheel_callback)
+        LIB_SYMBOL(set_buffer_size_callback)
+        LIB_SYMBOL(set_sample_rate_callback)
+        LIB_SYMBOL(set_client_registration_callback)
+        LIB_SYMBOL(set_port_registration_callback)
+        LIB_SYMBOL(set_port_rename_callback)
+        LIB_SYMBOL(set_port_connect_callback)
+        LIB_SYMBOL(set_graph_order_callback)
+        LIB_SYMBOL(set_xrun_callback)
+        LIB_SYMBOL(set_latency_callback)
+
+        LIB_SYMBOL(set_freewheel)
+        LIB_SYMBOL(set_buffer_size)
+
+        LIB_SYMBOL(get_sample_rate)
+        LIB_SYMBOL(get_buffer_size)
+        LIB_SYMBOL(cpu_load)
+
+        LIB_SYMBOL(port_register)
+        LIB_SYMBOL(port_unregister)
+        LIB_SYMBOL(port_get_buffer)
+
+        LIB_SYMBOL(port_name)
+        LIB_SYMBOL(port_uuid)
+        LIB_SYMBOL(port_short_name)
+        LIB_SYMBOL(port_flags)
+        LIB_SYMBOL(port_type)
+        LIB_SYMBOL(port_is_mine)
+        LIB_SYMBOL(port_connected)
+        LIB_SYMBOL(port_connected_to)
+        LIB_SYMBOL(port_get_connections)
+        LIB_SYMBOL(port_get_all_connections)
+
+        LIB_SYMBOL(port_rename)
+        LIB_SYMBOL(port_set_name)
+        LIB_SYMBOL(port_set_alias)
+        LIB_SYMBOL(port_unset_alias)
+        LIB_SYMBOL(port_get_aliases)
+
+        LIB_SYMBOL(port_request_monitor)
+        LIB_SYMBOL(port_request_monitor_by_name)
+        LIB_SYMBOL(port_ensure_monitor)
+        LIB_SYMBOL(port_monitoring_input)
+
+        LIB_SYMBOL(connect)
+        LIB_SYMBOL(disconnect)
+        LIB_SYMBOL(port_disconnect)
+
+        LIB_SYMBOL(port_name_size)
+        LIB_SYMBOL(port_type_size)
+        LIB_SYMBOL(port_type_get_buffer_size)
+
+        LIB_SYMBOL(port_get_latency_range)
+        LIB_SYMBOL(port_set_latency_range)
+        LIB_SYMBOL(recompute_total_latencies)
+
+        LIB_SYMBOL(get_ports)
+        LIB_SYMBOL(port_by_name)
+        LIB_SYMBOL(port_by_id)
+
+        LIB_SYMBOL(free)
+
+        LIB_SYMBOL(midi_get_event_count)
+        LIB_SYMBOL(midi_event_get)
+        LIB_SYMBOL(midi_clear_buffer)
+        LIB_SYMBOL(midi_event_write)
+        LIB_SYMBOL(midi_event_reserve)
+
+        LIB_SYMBOL(release_timebase)
+        LIB_SYMBOL(set_sync_callback)
+        LIB_SYMBOL(set_sync_timeout)
+        LIB_SYMBOL(set_timebase_callback)
+        LIB_SYMBOL(transport_locate)
+        LIB_SYMBOL(transport_query)
+        LIB_SYMBOL(get_current_transport_frame)
+        LIB_SYMBOL(transport_reposition)
+        LIB_SYMBOL(transport_start)
+        LIB_SYMBOL(transport_stop)
+
+        LIB_SYMBOL(set_property)
+        LIB_SYMBOL(get_property)
+        LIB_SYMBOL(free_description)
+        LIB_SYMBOL(get_properties)
+        LIB_SYMBOL(get_all_properties)
+        LIB_SYMBOL(remove_property)
+        LIB_SYMBOL(remove_properties)
+        LIB_SYMBOL(remove_all_properties)
+        LIB_SYMBOL(set_property_change_callback)
+
+        LIB_SYMBOL(set_process_thread)
+        LIB_SYMBOL(cycle_wait)
+        LIB_SYMBOL(cycle_signal)
+
+       #ifdef __WINE__
+        LIB_SYMBOL(set_thread_creator)
+       #endif
+      #endif
+
+        #undef JOIN
+        #undef LIB_SYMBOL
+    }
+
+   #ifdef HAVE_JACK
+    ~JackBridge() noexcept
+    {
+        USE_NAMESPACE_DISTRHO
+
+        if (lib != nullptr)
+        {
+            lib_close(lib);
+            lib = nullptr;
+        }
+    }
+   #endif
+
+    DISTRHO_DECLARE_NON_COPYABLE(JackBridge);
+};
+
+static bool usingNativeBridge = false;
+static bool usingRealJACK = true;
+static NativeBridge* nativeBridge = nullptr;
+
+// -----------------------------------------------------------------------------
+
+static JackBridge& getBridgeInstance() noexcept
+{
+    static JackBridge bridge;
+    return bridge;
+}
+
+#endif // ! (defined(JACKBRIDGE_DIRECT) || defined(JACKBRIDGE_DUMMY))
+
+// -----------------------------------------------------------------------------
+
+#if defined(__WINE__) && ! defined(JACKBRIDGE_DIRECT)
+
+struct WineBridge {
+    void* ptr;
+    JackLatencyCallback latency_cb;
+    JackProcessCallback process_cb;
+    JackThreadInitCallback thread_init_cb;
+    JackGraphOrderCallback graph_order_cb;
+    JackXRunCallback xrun_cb;
+    JackBufferSizeCallback bufsize_cb;
+    JackSampleRateCallback srate_cb;
+    JackPortRegistrationCallback port_reg_cb;
+    JackClientRegistrationCallback client_reg_cb;
+    JackPortConnectCallback port_conn_cb;
+    JackPortRenameCallback port_rename_cb;
+    JackFreewheelCallback freewheel_cb;
+    JackShutdownCallback shutdown_cb;
+    JackInfoShutdownCallback info_shutdown_cb;
+    JackSyncCallback sync_cb;
+    JackTimebaseCallback timebase_cb;
+    JackSessionCallback session_cb;
+    JackPropertyChangeCallback prop_change_cb;
+    JackThreadCallback proc_thread_cb;
+
+    void* (*creator_func)(void*);
+    void* creator_arg;
+    HANDLE creator_handle;
+    pthread_t creator_pthread;
+
+    WineBridge() noexcept
+        : ptr(nullptr),
+          latency_cb(nullptr),
+          process_cb(nullptr),
+          thread_init_cb(nullptr),
+          graph_order_cb(nullptr),
+          xrun_cb(nullptr),
+          bufsize_cb(nullptr),
+          srate_cb(nullptr),
+          port_reg_cb(nullptr),
+          client_reg_cb(nullptr),
+          port_conn_cb(nullptr),
+          port_rename_cb(nullptr),
+          freewheel_cb(nullptr),
+          shutdown_cb(nullptr),
+          info_shutdown_cb(nullptr),
+          sync_cb(nullptr),
+          timebase_cb(nullptr),
+          session_cb(nullptr),
+          prop_change_cb(nullptr),
+          proc_thread_cb(nullptr),
+          creator_func(nullptr),
+          creator_arg(nullptr),
+          creator_handle(nullptr),
+          creator_pthread(0) {}
+
+    static WineBridge& getInstance() noexcept
+    {
+        static WineBridge bridge;
+        return bridge;
+    }
+
+    void set_latency       (JackLatencyCallback            cb) noexcept { latency_cb       = cb; }
+    void set_process       (JackProcessCallback            cb) noexcept { process_cb       = cb; }
+    void set_thread_init   (JackThreadInitCallback         cb) noexcept { thread_init_cb   = cb; }
+    void set_graph_order   (JackGraphOrderCallback         cb) noexcept { graph_order_cb   = cb; }
+    void set_xrun          (JackXRunCallback               cb) noexcept { xrun_cb          = cb; }
+    void set_bufsize       (JackBufferSizeCallback         cb) noexcept { bufsize_cb       = cb; }
+    void set_srate         (JackSampleRateCallback         cb) noexcept { srate_cb         = cb; }
+    void set_port_reg      (JackPortRegistrationCallback   cb) noexcept { port_reg_cb      = cb; }
+    void set_client_reg    (JackClientRegistrationCallback cb) noexcept { client_reg_cb    = cb; }
+    void set_port_conn     (JackPortConnectCallback        cb) noexcept { port_conn_cb     = cb; }
+    void set_port_rename   (JackPortRenameCallback         cb) noexcept { port_rename_cb   = cb; }
+    void set_freewheel     (JackFreewheelCallback          cb) noexcept { freewheel_cb     = cb; }
+    void set_shutdown      (JackShutdownCallback           cb) noexcept { shutdown_cb      = cb; }
+    void set_info_shutdown (JackInfoShutdownCallback       cb) noexcept { info_shutdown_cb = cb; }
+    void set_sync          (JackSyncCallback               cb) noexcept { sync_cb          = cb; }
+    void set_timebase      (JackTimebaseCallback           cb) noexcept { timebase_cb      = cb; }
+    void set_session       (JackSessionCallback            cb) noexcept { session_cb       = cb; }
+    void set_prop_change   (JackPropertyChangeCallback     cb) noexcept { prop_change_cb   = cb; }
+    void set_process_thread(JackThreadCallback             cb) noexcept { proc_thread_cb   = cb; }
+
+    static DWORD WINAPI thread_creator_helper(LPVOID)
+    {
+        WineBridge& inst(getInstance());
+
+        inst.creator_pthread = pthread_self();
+        SetEvent(inst.creator_handle);
+        inst.creator_func(inst.creator_arg);
+        return 0;
+    }
+
+    static int thread_creator(pthread_t* thread_id, const pthread_attr_t*, void *(*function)(void*), void* arg)
+    {
+        WineBridge& inst(getInstance());
+
+        inst.creator_func   = function;
+        inst.creator_arg    = arg;
+        inst.creator_handle = ::CreateEventW(nullptr, false, false, nullptr);
+
+#if 0
+        ::CreateThread(nullptr, 0, thread_creator_helper, arg, 0, nullptr);
+#else
+        HANDLE handle = ::CreateThread(nullptr, 0, thread_creator_helper, arg, CREATE_SUSPENDED, nullptr);
+
+        if (handle == INVALID_HANDLE_VALUE)
+            return 1;
+
+        // TODO read attrs and decide this
+        ::SetThreadPriority(handle, THREAD_PRIORITY_TIME_CRITICAL);
+        ::ResumeThread(handle);
+#endif
+
+        ::WaitForSingleObject(inst.creator_handle, INFINITE);
+        *thread_id = inst.creator_pthread;
+        return 0;
+    }
+
+    static void latency(jack_latency_callback_mode_t mode, void* arg)
+    {
+        return getInstance().latency_cb(mode, arg);
+    }
+
+    static int process(jack_nframes_t nframes, void* arg)
+    {
+        return getInstance().process_cb(nframes, arg);
+    }
+
+    static void thread_init(void* arg)
+    {
+        return getInstance().thread_init_cb(arg);
+    }
+
+    static int graph_order(void* arg)
+    {
+        return getInstance().graph_order_cb(arg);
+    }
+
+    static int xrun(void* arg)
+    {
+        return getInstance().xrun_cb(arg);
+    }
+
+    static int bufsize(jack_nframes_t nframes, void* arg)
+    {
+        return getInstance().bufsize_cb(nframes, arg);
+    }
+
+    static int srate(jack_nframes_t nframes, void* arg)
+    {
+        return getInstance().srate_cb(nframes, arg);
+    }
+
+    static void port_reg(jack_port_id_t port, int register_, void* arg)
+    {
+        return getInstance().port_reg_cb(port, register_, arg);
+    }
+
+    static void client_reg(const char* name, int register_, void* arg)
+    {
+        return getInstance().client_reg_cb(name, register_, arg);
+    }
+
+    static void port_conn(jack_port_id_t a, jack_port_id_t b, int connect, void* arg)
+    {
+        return getInstance().port_conn_cb(a, b, connect, arg);
+    }
+
+    static void port_rename(jack_port_id_t port, const char* old_name, const char* new_name, void* arg)
+    {
+        getInstance().port_rename_cb(port, old_name, new_name, arg);
+    }
+
+    static void freewheel(int starting, void* arg)
+    {
+        return getInstance().freewheel_cb(starting, arg);
+    }
+
+    static void shutdown(void* arg)
+    {
+        return getInstance().shutdown_cb(arg);
+    }
+
+    static void info_shutdown(jack_status_t code, const char* reason, void* arg)
+    {
+        return getInstance().info_shutdown_cb(code, reason, arg);
+    }
+
+    static int sync(jack_transport_state_t state, jack_position_t* pos, void* arg)
+    {
+        return getInstance().sync_cb(state, pos, arg);
+    }
+
+    static void timebase(jack_transport_state_t state, jack_nframes_t nframes, jack_position_t* pos, int new_pos, void* arg)
+    {
+        return getInstance().timebase_cb(state, nframes, pos, new_pos, arg);
+    }
+
+    static void session(jack_session_event_t* event, void* arg)
+    {
+        return getInstance().session_cb(event, arg);
+    }
+
+    static void prop_change(jack_uuid_t subject, const char* key, jack_property_change_t change, void* arg)
+    {
+        return getInstance().prop_change_cb(subject, key, change, arg);
+    }
+
+    static void* process_thread(void* arg)
+    {
+        return getInstance().proc_thread_cb(arg);
+    }
+
+    DISTRHO_DECLARE_NON_COPYABLE(WineBridge);
+};
+
+#endif // __WINE__ && ! JACKBRIDGE_DIRECT
+
+// -----------------------------------------------------------------------------
+
+bool jackbridge_is_ok() noexcept
+{
+#if defined(JACKBRIDGE_DUMMY)
+    return false;
+#elif defined(JACKBRIDGE_DIRECT) || defined(DISTRHO_OS_WASM) || defined(RTAUDIO_API_TYPE)
+    return true;
+#else
+    return (getBridgeInstance().lib != nullptr);
+#endif
+}
+
+void jackbridge_init()
+{
+#if defined(__WINE__) && !defined(JACKBRIDGE_DIRECT)
+    if (getBridgeInstance().set_thread_creator_ptr != nullptr)
+        getBridgeInstance().set_thread_creator_ptr(WineBridge::thread_creator);
+#endif
+}
+
+// -----------------------------------------------------------------------------
+
+void jackbridge_get_version(int* major_ptr, int* minor_ptr, int* micro_ptr, int* proto_ptr)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_get_version(major_ptr, minor_ptr, micro_ptr, proto_ptr);
+#else
+    if (usingRealJACK && getBridgeInstance().get_version_ptr != nullptr)
+        return getBridgeInstance().get_version_ptr(major_ptr, minor_ptr, micro_ptr, proto_ptr);
+#endif
+    if (major_ptr != nullptr)
+        *major_ptr = 0;
+    if (minor_ptr != nullptr)
+        *minor_ptr = 0;
+    if (micro_ptr != nullptr)
+        *micro_ptr = 0;
+    if (proto_ptr != nullptr)
+        *proto_ptr = 0;
+}
+
+const char* jackbridge_get_version_string()
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_get_version_string();
+#else
+    if (usingRealJACK && getBridgeInstance().get_version_string_ptr != nullptr)
+        return getBridgeInstance().get_version_string_ptr();
+#endif
+    return nullptr;
+}
+
+// -----------------------------------------------------------------------------
+
+jack_client_t* jackbridge_client_open(const char* client_name, uint32_t options, jack_status_t* status)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_client_open(client_name, static_cast<jack_options_t>(options), status);
+#else
+   #ifndef DISTRHO_OS_WASM
+    if (getBridgeInstance().client_open_ptr != nullptr)
+        if (jack_client_t* const client = getBridgeInstance().client_open_ptr(client_name, static_cast<jack_options_t>(options), status))
+            return client;
+   #endif
+
+    static jack_client_t* const kValidClient = (jack_client_t*)0x1;
+
+    // maybe unused
+    (void)kValidClient;
+
+    usingNativeBridge = true;
+    usingRealJACK = false;
+
+   #ifdef DISTRHO_OS_WASM
+    nativeBridge = new WebBridge;
+    if (nativeBridge->open(client_name))
+        return kValidClient;
+    delete nativeBridge;
+   #endif
+    
+   #if defined(HAVE_RTAUDIO) && defined(RTAUDIO_API_TYPE)
+    nativeBridge = new RtAudioBridge;
+    if (nativeBridge->open(client_name))
+        return kValidClient;
+    delete nativeBridge;
+   #endif
+
+   #if defined(HAVE_SDL2) && DISTRHO_PLUGIN_NUM_INPUTS+DISTRHO_PLUGIN_NUM_OUTPUTS > 0
+    nativeBridge = new SDL2Bridge;
+    if (nativeBridge->open(client_name))
+        return kValidClient;
+    delete nativeBridge;
+   #endif
+#endif
+
+    if (status != nullptr)
+    {
+        int err = JackServerError;
+
+       #if !(defined(JACKBRIDGE_DUMMY) || defined(JACKBRIDGE_DIRECT))
+        if (nativeBridge != nullptr)
+        {
+            err
+            #ifdef HAVE_JACK
+             |=
+            #else
+             =
+            #endif
+            JackBridgeNativeFailed;
+        }
+       #endif
+
+        *status = static_cast<jack_status_t>(err);
+    }
+
+    return nullptr;
+}
+
+bool jackbridge_client_close(jack_client_t* client)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_client_close(client) == 0);
+#else
+    if (usingNativeBridge)
+    {
+        if (nativeBridge != nullptr)
+        {
+            nativeBridge->close();
+            delete nativeBridge;
+            nativeBridge = nullptr;
+        }
+        usingNativeBridge = false;
+        usingRealJACK = true;
+        return true;
+    }
+    if (getBridgeInstance().client_close_ptr != nullptr)
+        return (getBridgeInstance().client_close_ptr(client) == 0);
+#endif
+    return false;
+}
+
+// -----------------------------------------------------------------------------
+
+int jackbridge_client_name_size()
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_client_name_size();
+#else
+    if (usingRealJACK && getBridgeInstance().client_name_size_ptr != nullptr)
+        return getBridgeInstance().client_name_size_ptr();
+#endif
+    return 33;
+}
+
+const char* jackbridge_get_client_name(jack_client_t* client)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_get_client_name(client);
+#else
+    if (usingNativeBridge)
+        return DISTRHO_PLUGIN_NAME;
+    if (getBridgeInstance().get_client_name_ptr != nullptr)
+        return getBridgeInstance().get_client_name_ptr(client);
+#endif
+    return nullptr;
+}
+
+// -----------------------------------------------------------------------------
+
+char* jackbridge_client_get_uuid(jack_client_t* client)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_client_get_uuid(client);
+#else
+    if (usingRealJACK)
+        if (const jacksym_client_get_uuid func = getBridgeInstance().client_get_uuid_ptr)
+            return func(client);
+#endif
+    return nullptr;
+}
+
+char* jackbridge_get_uuid_for_client_name(jack_client_t* client, const char* name)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_get_uuid_for_client_name(client, name);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().get_uuid_for_client_name_ptr != nullptr)
+            return getBridgeInstance().get_uuid_for_client_name_ptr(client, name);
+#endif
+    return nullptr;
+}
+
+char* jackbridge_get_client_name_by_uuid(jack_client_t* client, const char* uuid)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_get_client_name_by_uuid(client, uuid);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().get_client_name_by_uuid_ptr != nullptr)
+            return getBridgeInstance().get_client_name_by_uuid_ptr(client, uuid);
+#endif
+    return nullptr;
+}
+
+// -----------------------------------------------------------------------------
+
+bool jackbridge_uuid_parse(const char* buf, jack_uuid_t* uuid)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_uuid_parse(buf, uuid) == 0);
+#else
+    if (usingRealJACK)
+        if (const jacksym_uuid_parse func = getBridgeInstance().uuid_parse_ptr)
+            return (func(buf, uuid) == 0);
+#endif
+    return false;
+}
+
+void jackbridge_uuid_unparse(jack_uuid_t uuid, char buf[JACK_UUID_STRING_SIZE])
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    jack_uuid_unparse(uuid, buf);
+#else
+    if (usingRealJACK)
+        if (const jacksym_uuid_unparse func = getBridgeInstance().uuid_unparse_ptr)
+            return func(uuid, buf);
+#endif
+}
+
+// -----------------------------------------------------------------------------
+
+bool jackbridge_activate(jack_client_t* client)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_activate(client) == 0);
+#else
+    if (usingNativeBridge)
+        return nativeBridge->activate();
+    if (getBridgeInstance().activate_ptr != nullptr)
+        return (getBridgeInstance().activate_ptr(client) == 0);
+#endif
+    return false;
+}
+
+bool jackbridge_deactivate(jack_client_t* client)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_deactivate(client) == 0);
+#else
+    if (usingNativeBridge)
+        return nativeBridge->deactivate();
+    if (getBridgeInstance().deactivate_ptr != nullptr)
+        return (getBridgeInstance().deactivate_ptr(client) == 0);
+#endif
+    return false;
+}
+
+bool jackbridge_is_realtime(jack_client_t* client)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_is_realtime(client);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().is_realtime_ptr != nullptr)
+            return getBridgeInstance().is_realtime_ptr(client);
+#endif
+    return false;
+}
+
+// -----------------------------------------------------------------------------
+
+bool jackbridge_set_thread_init_callback(jack_client_t* client, JackThreadInitCallback thread_init_callback, void* arg)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_set_thread_init_callback(client, thread_init_callback, arg) == 0);
+#else
+    if (usingRealJACK && getBridgeInstance().set_thread_init_callback_ptr != nullptr)
+    {
+# ifdef __WINE__
+        WineBridge::getInstance().set_thread_init(thread_init_callback);
+        return (getBridgeInstance().set_thread_init_callback_ptr(client, WineBridge::thread_init, arg) == 0);
+# else
+        return (getBridgeInstance().set_thread_init_callback_ptr(client, thread_init_callback, arg) == 0);
+# endif
+    }
+#endif
+    return false;
+}
+
+void jackbridge_on_shutdown(jack_client_t* client, JackShutdownCallback shutdown_callback, void* arg)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    jack_on_shutdown(client, shutdown_callback, arg);
+#else
+    if (usingRealJACK && getBridgeInstance().on_shutdown_ptr != nullptr)
+    {
+# ifdef __WINE__
+        WineBridge::getInstance().set_shutdown(shutdown_callback);
+        getBridgeInstance().on_shutdown_ptr(client, WineBridge::shutdown, arg);
+# else
+        getBridgeInstance().on_shutdown_ptr(client, shutdown_callback, arg);
+# endif
+    }
+#endif
+}
+
+void jackbridge_on_info_shutdown(jack_client_t* client, JackInfoShutdownCallback shutdown_callback, void* arg)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    jack_on_info_shutdown(client, shutdown_callback, arg);
+#else
+    if (usingRealJACK && getBridgeInstance().on_info_shutdown_ptr != nullptr)
+    {
+# ifdef __WINE__
+        WineBridge::getInstance().set_info_shutdown(shutdown_callback);
+        getBridgeInstance().on_info_shutdown_ptr(client, WineBridge::info_shutdown, arg);
+# else
+        getBridgeInstance().on_info_shutdown_ptr(client, shutdown_callback, arg);
+# endif
+    }
+#endif
+}
+
+bool jackbridge_set_process_callback(jack_client_t* client, JackProcessCallback process_callback, void* arg)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_set_process_callback(client, process_callback, arg) == 0);
+#else
+    if (usingNativeBridge)
+    {
+        nativeBridge->jackProcessCallback = process_callback;
+        nativeBridge->jackProcessArg = arg;
+        return true;
+    }
+    if (getBridgeInstance().set_process_callback_ptr != nullptr)
+    {
+# ifdef __WINE__
+        WineBridge::getInstance().set_process(process_callback);
+        return (getBridgeInstance().set_process_callback_ptr(client, WineBridge::process, arg) == 0);
+# else
+        return (getBridgeInstance().set_process_callback_ptr(client, process_callback, arg) == 0);
+# endif
+    }
+#endif
+    return false;
+}
+
+bool jackbridge_set_freewheel_callback(jack_client_t* client, JackFreewheelCallback freewheel_callback, void* arg)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_set_freewheel_callback(client, freewheel_callback, arg) == 0);
+#else
+    if (usingRealJACK && getBridgeInstance().set_freewheel_callback_ptr != nullptr)
+    {
+# ifdef __WINE__
+        WineBridge::getInstance().set_freewheel(freewheel_callback);
+        return (getBridgeInstance().set_freewheel_callback_ptr(client, WineBridge::freewheel, arg) == 0);
+# else
+        return (getBridgeInstance().set_freewheel_callback_ptr(client, freewheel_callback, arg) == 0);
+# endif
+    }
+#endif
+    return false;
+}
+
+bool jackbridge_set_buffer_size_callback(jack_client_t* client, JackBufferSizeCallback bufsize_callback, void* arg)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_set_buffer_size_callback(client, bufsize_callback, arg) == 0);
+#else
+    if (usingNativeBridge)
+    {
+        nativeBridge->bufferSizeCallback = bufsize_callback;
+        nativeBridge->jackBufferSizeArg = arg;
+        return true;
+    }
+    if (getBridgeInstance().set_buffer_size_callback_ptr != nullptr)
+    {
+# ifdef __WINE__
+        WineBridge::getInstance().set_bufsize(bufsize_callback);
+        return (getBridgeInstance().set_buffer_size_callback_ptr(client, WineBridge::bufsize, arg) == 0);
+# else
+        return (getBridgeInstance().set_buffer_size_callback_ptr(client, bufsize_callback, arg) == 0);
+# endif
+    }
+#endif
+    return false;
+}
+
+bool jackbridge_set_sample_rate_callback(jack_client_t* client, JackSampleRateCallback srate_callback, void* arg)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_set_sample_rate_callback(client, srate_callback, arg) == 0);
+#else
+    if (usingRealJACK && getBridgeInstance().set_sample_rate_callback_ptr != nullptr)
+    {
+# ifdef __WINE__
+        WineBridge::getInstance().set_srate(srate_callback);
+        return (getBridgeInstance().set_sample_rate_callback_ptr(client, WineBridge::srate, arg) == 0);
+# else
+        return (getBridgeInstance().set_sample_rate_callback_ptr(client, srate_callback, arg) == 0);
+# endif
+    }
+#endif
+    return false;
+}
+
+bool jackbridge_set_client_registration_callback(jack_client_t* client, JackClientRegistrationCallback registration_callback, void* arg)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_set_client_registration_callback(client, registration_callback, arg) == 0);
+#else
+    if (usingRealJACK && getBridgeInstance().set_client_registration_callback_ptr != nullptr)
+    {
+# ifdef __WINE__
+        WineBridge::getInstance().set_client_reg(registration_callback);
+        return (getBridgeInstance().set_client_registration_callback_ptr(client, WineBridge::client_reg, arg) == 0);
+# else
+        return (getBridgeInstance().set_client_registration_callback_ptr(client, registration_callback, arg) == 0);
+# endif
+    }
+#endif
+    return false;
+}
+
+bool jackbridge_set_port_registration_callback(jack_client_t* client, JackPortRegistrationCallback registration_callback, void *arg)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_set_port_registration_callback(client, registration_callback, arg) == 0);
+#else
+    if (usingRealJACK && getBridgeInstance().set_port_registration_callback_ptr != nullptr)
+    {
+# ifdef __WINE__
+        WineBridge::getInstance().set_port_reg(registration_callback);
+        return (getBridgeInstance().set_port_registration_callback_ptr(client, WineBridge::port_reg, arg) == 0);
+# else
+        return (getBridgeInstance().set_port_registration_callback_ptr(client, registration_callback, arg) == 0);
+# endif
+    }
+#endif
+    return false;
+}
+
+bool jackbridge_set_port_rename_callback(jack_client_t* client, JackPortRenameCallback rename_callback, void* arg)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_set_port_rename_callback(client, rename_callback, arg) == 0);
+#else
+    if (usingRealJACK && getBridgeInstance().set_port_rename_callback_ptr != nullptr)
+    {
+# ifdef __WINE__
+        WineBridge::getInstance().set_port_rename(rename_callback);
+        return (getBridgeInstance().set_port_rename_callback_ptr(client, WineBridge::port_rename, arg) == 0);
+# else
+        return (getBridgeInstance().set_port_rename_callback_ptr(client, rename_callback, arg) == 0);
+# endif
+    }
+#endif
+    return false;
+}
+
+bool jackbridge_set_port_connect_callback(jack_client_t* client, JackPortConnectCallback connect_callback, void* arg)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_set_port_connect_callback(client, connect_callback, arg) == 0);
+#else
+    if (usingRealJACK && getBridgeInstance().set_port_connect_callback_ptr != nullptr)
+    {
+# ifdef __WINE__
+        WineBridge::getInstance().set_port_conn(connect_callback);
+        return (getBridgeInstance().set_port_connect_callback_ptr(client, WineBridge::port_conn, arg) == 0);
+# else
+        return (getBridgeInstance().set_port_connect_callback_ptr(client, connect_callback, arg) == 0);
+# endif
+    }
+#endif
+    return false;
+}
+
+bool jackbridge_set_graph_order_callback(jack_client_t* client, JackGraphOrderCallback graph_callback, void* arg)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_set_graph_order_callback(client, graph_callback, arg) == 0);
+#else
+    if (usingRealJACK && getBridgeInstance().set_graph_order_callback_ptr != nullptr)
+    {
+# ifdef __WINE__
+        WineBridge::getInstance().set_graph_order(graph_callback);
+        return (getBridgeInstance().set_graph_order_callback_ptr(client, WineBridge::graph_order, arg) == 0);
+# else
+        return (getBridgeInstance().set_graph_order_callback_ptr(client, graph_callback, arg) == 0);
+# endif
+    }
+#endif
+    return false;
+}
+
+bool jackbridge_set_xrun_callback(jack_client_t* client, JackXRunCallback xrun_callback, void* arg)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_set_xrun_callback(client, xrun_callback, arg) == 0);
+#else
+    if (usingRealJACK && getBridgeInstance().set_xrun_callback_ptr != nullptr)
+    {
+# ifdef __WINE__
+        WineBridge::getInstance().set_xrun(xrun_callback);
+        return (getBridgeInstance().set_xrun_callback_ptr(client, WineBridge::xrun, arg) == 0);
+# else
+        return (getBridgeInstance().set_xrun_callback_ptr(client, xrun_callback, arg) == 0);
+# endif
+    }
+#endif
+    return false;
+}
+
+bool jackbridge_set_latency_callback(jack_client_t* client, JackLatencyCallback latency_callback, void* arg)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_set_latency_callback(client, latency_callback, arg) == 0);
+#else
+    if (usingRealJACK && getBridgeInstance().set_latency_callback_ptr != nullptr)
+    {
+# ifdef __WINE__
+        WineBridge::getInstance().set_latency(latency_callback);
+        return (getBridgeInstance().set_latency_callback_ptr(client, WineBridge::latency, arg) == 0);
+# else
+        return (getBridgeInstance().set_latency_callback_ptr(client, latency_callback, arg) == 0);
+# endif
+    }
+#endif
+    return false;
+}
+
+// -----------------------------------------------------------------------------
+
+bool jackbridge_set_freewheel(jack_client_t* client, bool onoff)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_set_freewheel(client, onoff);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().set_freewheel_ptr != nullptr)
+            return getBridgeInstance().set_freewheel_ptr(client, onoff);
+#endif
+    return false;
+}
+
+bool jackbridge_set_buffer_size(jack_client_t* client, jack_nframes_t nframes)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_set_buffer_size(client, nframes);
+#else
+    if (usingNativeBridge)
+        return nativeBridge->requestBufferSizeChange(nframes);
+    if (getBridgeInstance().set_buffer_size_ptr != nullptr)
+        return getBridgeInstance().set_buffer_size_ptr(client, nframes);
+#endif
+    return false;
+}
+
+// -----------------------------------------------------------------------------
+
+jack_nframes_t jackbridge_get_sample_rate(jack_client_t* client)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_get_sample_rate(client);
+#else
+    if (usingNativeBridge)
+        return nativeBridge->sampleRate;
+    if (getBridgeInstance().get_sample_rate_ptr != nullptr)
+        return getBridgeInstance().get_sample_rate_ptr(client);
+#endif
+    return 0;
+}
+
+jack_nframes_t jackbridge_get_buffer_size(jack_client_t* client)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_get_buffer_size(client);
+#else
+    if (usingNativeBridge)
+        return nativeBridge->bufferSize;
+    if (getBridgeInstance().get_buffer_size_ptr != nullptr)
+        return getBridgeInstance().get_buffer_size_ptr(client);
+#endif
+    return 0;
+}
+
+float jackbridge_cpu_load(jack_client_t* client)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_cpu_load(client);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().cpu_load_ptr != nullptr)
+            return getBridgeInstance().cpu_load_ptr(client);
+#endif
+    return 0.0f;
+}
+
+// -----------------------------------------------------------------------------
+
+jack_port_t* jackbridge_port_register(jack_client_t* client, const char* port_name, const char* type, uint64_t flags, uint64_t buffer_size)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_port_register(client, port_name, type, flags, buffer_size);
+#else
+    if (usingNativeBridge)
+        return nativeBridge->registerPort(type, flags);
+    if (getBridgeInstance().port_register_ptr != nullptr)
+        return getBridgeInstance().port_register_ptr(client, port_name, type,
+                                                     static_cast<ulong>(flags),
+                                                     static_cast<ulong>(buffer_size));
+#endif
+    return nullptr;
+}
+
+bool jackbridge_port_unregister(jack_client_t* client, jack_port_t* port)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_port_unregister(client, port) == 0);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_unregister_ptr != nullptr)
+            return (getBridgeInstance().port_unregister_ptr(client, port) == 0);
+#endif
+    return false;
+}
+
+void* jackbridge_port_get_buffer(jack_port_t* port, jack_nframes_t nframes)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_port_get_buffer(port, nframes);
+#else
+    if (usingNativeBridge)
+        return nativeBridge->getPortBuffer(port);
+    if (getBridgeInstance().port_get_buffer_ptr != nullptr)
+        return getBridgeInstance().port_get_buffer_ptr(port, nframes);
+#endif
+    return nullptr;
+}
+
+// -----------------------------------------------------------------------------
+
+const char* jackbridge_port_name(const jack_port_t* port)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_port_name(port);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_name_ptr != nullptr)
+            return getBridgeInstance().port_name_ptr(port);
+#endif
+    return nullptr;
+}
+
+jack_uuid_t jackbridge_port_uuid(const jack_port_t* port)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_port_uuid(port);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_uuid_ptr != nullptr)
+            return getBridgeInstance().port_uuid_ptr(port);
+#endif
+    return 0;
+}
+
+const char* jackbridge_port_short_name(const jack_port_t* port)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_port_short_name(port);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_short_name_ptr != nullptr)
+            return getBridgeInstance().port_short_name_ptr(port);
+#endif
+    return nullptr;
+}
+
+int jackbridge_port_flags(const jack_port_t* port)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_port_flags(port);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_flags_ptr != nullptr)
+            return getBridgeInstance().port_flags_ptr(port);
+#endif
+    return 0x0;
+}
+
+const char* jackbridge_port_type(const jack_port_t* port)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_port_type(port);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_type_ptr != nullptr)
+            return getBridgeInstance().port_type_ptr(port);
+#endif
+    return nullptr;
+}
+
+bool jackbridge_port_is_mine(const jack_client_t* client, const jack_port_t* port)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_port_is_mine(client, port);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_is_mine_ptr != nullptr)
+            return getBridgeInstance().port_is_mine_ptr(client, port);
+#endif
+    return false;
+}
+
+int jackbridge_port_connected(const jack_port_t* port)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_port_connected(port);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_connected_ptr != nullptr)
+            return getBridgeInstance().port_connected_ptr(port);
+#endif
+    return 0;
+}
+
+bool jackbridge_port_connected_to(const jack_port_t* port, const char* port_name)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_port_connected_to(port, port_name);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_connected_to_ptr != nullptr)
+            return getBridgeInstance().port_connected_to_ptr(port, port_name);
+#endif
+    return false;
+}
+
+const char** jackbridge_port_get_connections(const jack_port_t* port)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_port_get_connections(port);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_get_connections_ptr != nullptr)
+            return getBridgeInstance().port_get_connections_ptr(port);
+#endif
+    return nullptr;
+}
+
+const char** jackbridge_port_get_all_connections(const jack_client_t* client, const jack_port_t* port)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_port_get_all_connections(client, port);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_get_all_connections_ptr != nullptr)
+            return getBridgeInstance().port_get_all_connections_ptr(client, port);
+#endif
+    return nullptr;
+}
+
+// -----------------------------------------------------------------------------
+
+bool jackbridge_port_rename(jack_client_t* client, jack_port_t* port, const char* port_name)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_port_rename(client, port, port_name) == 0);
+#else
+    if (usingNativeBridge)
+        return false;
+    // Try new API first
+    if (getBridgeInstance().port_rename_ptr != nullptr)
+        return (getBridgeInstance().port_rename_ptr(client, port, port_name) == 0);
+    // Try old API if using JACK2
+    if (getBridgeInstance().get_version_string_ptr != nullptr && getBridgeInstance().port_set_name_ptr != nullptr)
+        return (getBridgeInstance().port_set_name_ptr(port, port_name) == 0);
+#endif
+    return false;
+}
+
+bool jackbridge_port_set_alias(jack_port_t* port, const char* alias)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_port_set_alias(port, alias) == 0);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_set_alias_ptr != nullptr)
+            return (getBridgeInstance().port_set_alias_ptr(port, alias) == 0);
+#endif
+    return false;
+}
+
+bool jackbridge_port_unset_alias(jack_port_t* port, const char* alias)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_port_unset_alias(port, alias) == 0);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_unset_alias_ptr != nullptr)
+            return (getBridgeInstance().port_unset_alias_ptr(port, alias) == 0);
+#endif
+    return false;
+}
+
+int jackbridge_port_get_aliases(const jack_port_t* port, char* const aliases[2])
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_port_get_aliases(port, aliases) == 0);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_get_aliases_ptr != nullptr)
+            return getBridgeInstance().port_get_aliases_ptr(port, aliases);
+#endif
+    return 0;
+}
+
+// -----------------------------------------------------------------------------
+
+bool jackbridge_port_request_monitor(jack_port_t* port, bool onoff)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_port_request_monitor(port, onoff) == 0);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_request_monitor_ptr != nullptr)
+            return (getBridgeInstance().port_request_monitor_ptr(port, onoff) == 0);
+#endif
+    return false;
+}
+
+bool jackbridge_port_request_monitor_by_name(jack_client_t* client, const char* port_name, bool onoff)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_port_request_monitor_by_name(client, port_name, onoff) == 0);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_request_monitor_by_name_ptr != nullptr)
+            return (getBridgeInstance().port_request_monitor_by_name_ptr(client, port_name, onoff) == 0);
+#endif
+    return false;
+}
+
+bool jackbridge_port_ensure_monitor(jack_port_t* port, bool onoff)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_port_ensure_monitor(port, onoff) == 0);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_ensure_monitor_ptr != nullptr)
+            return (getBridgeInstance().port_ensure_monitor_ptr(port, onoff) == 0);
+#endif
+    return false;
+}
+
+bool jackbridge_port_monitoring_input(jack_port_t* port)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_port_monitoring_input(port);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_monitoring_input_ptr != nullptr)
+            return getBridgeInstance().port_monitoring_input_ptr(port);
+#endif
+    return false;
+}
+
+// -----------------------------------------------------------------------------
+
+bool jackbridge_connect(jack_client_t* client, const char* source_port, const char* destination_port)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_connect(client, source_port, destination_port) == 0);
+#else
+    if (usingRealJACK && getBridgeInstance().connect_ptr != nullptr)
+    {
+        const int ret = getBridgeInstance().connect_ptr(client, source_port, destination_port);
+        return ret == 0 || ret == EEXIST;
+    }
+#endif
+    return false;
+}
+
+bool jackbridge_disconnect(jack_client_t* client, const char* source_port, const char* destination_port)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_disconnect(client, source_port, destination_port) == 0);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().disconnect_ptr != nullptr)
+            return (getBridgeInstance().disconnect_ptr(client, source_port, destination_port) == 0);
+#endif
+    return false;
+}
+
+bool jackbridge_port_disconnect(jack_client_t* client, jack_port_t* port)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_port_disconnect(client, port) == 0);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_disconnect_ptr != nullptr)
+            return (getBridgeInstance().port_disconnect_ptr(client, port) == 0);
+#endif
+    return false;
+}
+
+// -----------------------------------------------------------------------------
+
+int jackbridge_port_name_size()
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_port_name_size();
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_name_size_ptr != nullptr)
+            return getBridgeInstance().port_name_size_ptr();
+#endif
+    return 256;
+}
+
+int jackbridge_port_type_size()
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_port_type_size();
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_type_size_ptr != nullptr)
+            return getBridgeInstance().port_type_size_ptr();
+#endif
+    return 32;
+}
+
+uint32_t jackbridge_port_type_get_buffer_size(jack_client_t* client, const char* port_type)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return static_cast<uint32_t>(jack_port_type_get_buffer_size(client, port_type));
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_type_get_buffer_size_ptr != nullptr)
+            return static_cast<uint32_t>(getBridgeInstance().port_type_get_buffer_size_ptr(client, port_type));
+#endif
+    return 0;
+}
+
+// -----------------------------------------------------------------------------
+
+void jackbridge_port_get_latency_range(jack_port_t* port, uint32_t mode, jack_latency_range_t* range)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_port_get_latency_range(port, static_cast<jack_latency_callback_mode_t>(mode), range);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_get_latency_range_ptr != nullptr)
+            return getBridgeInstance().port_get_latency_range_ptr(port,
+                                                                  static_cast<jack_latency_callback_mode_t>(mode),
+                                                                  range);
+#endif
+    range->min = 0;
+    range->max = 0;
+}
+
+void jackbridge_port_set_latency_range(jack_port_t* port, uint32_t mode, jack_latency_range_t* range)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    jack_port_set_latency_range(port, static_cast<jack_latency_callback_mode_t>(mode), range);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_set_latency_range_ptr != nullptr)
+            getBridgeInstance().port_set_latency_range_ptr(port,
+                                                           static_cast<jack_latency_callback_mode_t>(mode),
+                                                           range);
+#endif
+}
+
+bool jackbridge_recompute_total_latencies(jack_client_t* client)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_recompute_total_latencies(client) == 0);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().recompute_total_latencies_ptr != nullptr)
+            return (getBridgeInstance().recompute_total_latencies_ptr(client) == 0);
+#endif
+    return false;
+}
+
+// -----------------------------------------------------------------------------
+
+const char** jackbridge_get_ports(jack_client_t* client, const char* port_name_pattern, const char* type_name_pattern, uint64_t flags)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_get_ports(client, port_name_pattern, type_name_pattern, flags);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().get_ports_ptr != nullptr)
+            return getBridgeInstance().get_ports_ptr(client, port_name_pattern, type_name_pattern,
+                                                     static_cast<ulong>(flags));
+#endif
+    return nullptr;
+}
+
+jack_port_t* jackbridge_port_by_name(jack_client_t* client, const char* port_name)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_port_by_name(client, port_name);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_by_name_ptr != nullptr)
+            return getBridgeInstance().port_by_name_ptr(client, port_name);
+#endif
+    return nullptr;
+}
+
+jack_port_t* jackbridge_port_by_id(jack_client_t* client, jack_port_id_t port_id)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_port_by_id(client, port_id);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().port_by_id_ptr != nullptr)
+            return getBridgeInstance().port_by_id_ptr(client, port_id);
+#endif
+    return nullptr;
+}
+
+// -----------------------------------------------------------------------------
+
+void jackbridge_free(void* ptr)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_free(ptr);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().free_ptr != nullptr)
+            return getBridgeInstance().free_ptr(ptr);
+
+    // just in case
+    std::free(ptr);
+#endif
+}
+
+// -----------------------------------------------------------------------------
+
+uint32_t jackbridge_midi_get_event_count(void* port_buffer)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_midi_get_event_count(port_buffer);
+#else
+    if (usingNativeBridge)
+        return nativeBridge->getEventCount();
+    if (getBridgeInstance().midi_get_event_count_ptr != nullptr)
+        return getBridgeInstance().midi_get_event_count_ptr(port_buffer);
+#endif
+    return 0;
+}
+
+bool jackbridge_midi_event_get(jack_midi_event_t* event, void* port_buffer, uint32_t event_index)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_midi_event_get(event, port_buffer, event_index) == 0);
+#else
+    if (usingNativeBridge)
+        return nativeBridge->getEvent(event);
+    if (getBridgeInstance().midi_event_get_ptr != nullptr)
+        return (getBridgeInstance().midi_event_get_ptr(event, port_buffer, event_index) == 0);
+#endif
+    return false;
+}
+
+void jackbridge_midi_clear_buffer(void* port_buffer)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    jack_midi_clear_buffer(port_buffer);
+#else
+    if (usingNativeBridge)
+        return nativeBridge->clearEventBuffer();
+    if (getBridgeInstance().midi_clear_buffer_ptr != nullptr)
+        getBridgeInstance().midi_clear_buffer_ptr(port_buffer);
+#endif
+}
+
+bool jackbridge_midi_event_write(void* port_buffer, jack_nframes_t time, const jack_midi_data_t* data, uint32_t data_size)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_midi_event_write(port_buffer, time, data, data_size) == 0);
+#else
+    if (usingNativeBridge)
+        return nativeBridge->writeEvent(time, data, data_size);
+    if (getBridgeInstance().midi_event_write_ptr != nullptr)
+        return (getBridgeInstance().midi_event_write_ptr(port_buffer, time, data, data_size) == 0);
+#endif
+    return false;
+}
+
+jack_midi_data_t* jackbridge_midi_event_reserve(void* port_buffer, jack_nframes_t time, uint32_t data_size)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_midi_event_reserve(port_buffer, time, data_size);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().midi_event_reserve_ptr != nullptr)
+            return getBridgeInstance().midi_event_reserve_ptr(port_buffer, time, data_size);
+#endif
+    return nullptr;
+}
+
+// -----------------------------------------------------------------------------
+
+bool jackbridge_release_timebase(jack_client_t* client)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_release_timebase(client) == 0);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().release_timebase_ptr != nullptr)
+            return (getBridgeInstance().release_timebase_ptr(client) == 0);
+#endif
+    return false;
+}
+
+bool jackbridge_set_sync_callback(jack_client_t* client, JackSyncCallback sync_callback, void* arg)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_set_sync_callback(client, sync_callback, arg) == 0);
+#else
+    if (usingRealJACK && getBridgeInstance().set_sync_callback_ptr != nullptr)
+    {
+# ifdef __WINE__
+        WineBridge::getInstance().set_sync(sync_callback);
+        return (getBridgeInstance().set_sync_callback_ptr(client, WineBridge::sync, arg) == 0);
+# else
+        return (getBridgeInstance().set_sync_callback_ptr(client, sync_callback, arg) == 0);
+# endif
+    }
+#endif
+    return false;
+}
+
+bool jackbridge_set_sync_timeout(jack_client_t* client, jack_time_t timeout)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_set_sync_timeout(client, timeout) == 0);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().set_sync_timeout_ptr != nullptr)
+            return (getBridgeInstance().set_sync_timeout_ptr(client, timeout) == 0);
+#endif
+    return false;
+}
+
+bool jackbridge_set_timebase_callback(jack_client_t* client, bool conditional, JackTimebaseCallback timebase_callback, void* arg)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_set_timebase_callback(client, conditional, timebase_callback, arg) == 0);
+#else
+    if (usingRealJACK && getBridgeInstance().set_timebase_callback_ptr != nullptr)
+    {
+# ifdef __WINE__
+        WineBridge::getInstance().set_timebase(timebase_callback);
+        return (getBridgeInstance().set_timebase_callback_ptr(client, conditional, WineBridge::timebase, arg) == 0);
+# else
+        return (getBridgeInstance().set_timebase_callback_ptr(client, conditional, timebase_callback, arg) == 0);
+# endif
+    }
+#endif
+    return false;
+}
+
+bool jackbridge_transport_locate(jack_client_t* client, jack_nframes_t frame)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_transport_locate(client, frame) == 0);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().transport_locate_ptr != nullptr)
+            return (getBridgeInstance().transport_locate_ptr(client, frame) == 0);
+#endif
+    return false;
+}
+
+uint32_t jackbridge_transport_query(const jack_client_t* client, jack_position_t* pos)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_transport_query(client, pos);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().transport_query_ptr != nullptr)
+            return getBridgeInstance().transport_query_ptr(client, pos);
+#endif
+    if (pos != nullptr)
+    {
+        // invalidate
+        std::memset(pos, 0, sizeof(*pos));
+        pos->unique_1 = 0;
+        pos->unique_2 = 1;
+    }
+    return JackTransportStopped;
+}
+
+jack_nframes_t jackbridge_get_current_transport_frame(const jack_client_t* client)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_get_current_transport_frame(client);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().get_current_transport_frame_ptr != nullptr)
+            return getBridgeInstance().get_current_transport_frame_ptr(client);
+#endif
+    return 0;
+}
+
+bool jackbridge_transport_reposition(jack_client_t* client, const jack_position_t* pos)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_transport_reposition(client, pos) == 0);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().transport_reposition_ptr != nullptr)
+            return (getBridgeInstance().transport_reposition_ptr(client, pos) == 0);
+#endif
+    return false;
+}
+
+void jackbridge_transport_start(jack_client_t* client)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    jack_transport_start(client);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().transport_start_ptr != nullptr)
+            getBridgeInstance().transport_start_ptr(client);
+#endif
+}
+
+void jackbridge_transport_stop(jack_client_t* client)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    jack_transport_stop(client);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().transport_stop_ptr != nullptr)
+            getBridgeInstance().transport_stop_ptr(client);
+#endif
+}
+
+// -----------------------------------------------------------------------------
+
+bool jackbridge_set_property(jack_client_t* client, jack_uuid_t subject, const char* key, const char* value, const char* type)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_set_property(client, subject, key, value, type) == 0);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().set_property_ptr != nullptr)
+            return (getBridgeInstance().set_property_ptr(client, subject, key, value, type) == 0);
+#endif
+    return false;
+}
+
+bool jackbridge_get_property(jack_uuid_t subject, const char* key, char** value, char** type)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_get_property(subject, key, value, type) == 0);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().get_property_ptr != nullptr)
+            return (getBridgeInstance().get_property_ptr(subject, key, value, type) == 0);
+#endif
+    return false;
+}
+
+void jackbridge_free_description(jack_description_t* desc, bool free_description_itself)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    jack_free_description(desc, free_description_itself);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().free_description_ptr != nullptr)
+            getBridgeInstance().free_description_ptr(desc, free_description_itself);
+#endif
+}
+
+bool jackbridge_get_properties(jack_uuid_t subject, jack_description_t* desc)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_get_properties(subject, desc) == 0);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().get_properties_ptr != nullptr)
+            return (getBridgeInstance().get_properties_ptr(subject, desc) == 0);
+#endif
+    return false;
+}
+
+bool jackbridge_get_all_properties(jack_description_t** descs)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_get_all_properties(descs) == 0);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().get_all_properties_ptr != nullptr)
+            return (getBridgeInstance().get_all_properties_ptr(descs) == 0);
+#endif
+    return false;
+}
+
+bool jackbridge_remove_property(jack_client_t* client, jack_uuid_t subject, const char* key)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_remove_property(client, subject, key) == 0);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().remove_property_ptr != nullptr)
+            return (getBridgeInstance().remove_property_ptr(client, subject, key) == 0);
+#endif
+    return false;
+}
+
+int jackbridge_remove_properties(jack_client_t* client, jack_uuid_t subject)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_remove_properties(client, subject);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().remove_properties_ptr != nullptr)
+            return getBridgeInstance().remove_properties_ptr(client, subject);
+#endif
+    return 0;
+}
+
+bool jackbridge_remove_all_properties(jack_client_t* client)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_remove_all_properties(client) == 0);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().remove_all_properties_ptr != nullptr)
+            return (getBridgeInstance().remove_all_properties_ptr(client) == 0);
+#endif
+    return false;
+}
+
+bool jackbridge_set_property_change_callback(jack_client_t* client, JackPropertyChangeCallback callback, void* arg)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_set_property_change_callback(client, callback, arg) == 0);
+#else
+    if (usingRealJACK && getBridgeInstance().set_property_change_callback_ptr != nullptr)
+    {
+# ifdef __WINE__
+        WineBridge::getInstance().set_prop_change(callback);
+        return (getBridgeInstance().set_property_change_callback_ptr(client, WineBridge::prop_change, arg) == 0);
+# else
+        return (getBridgeInstance().set_property_change_callback_ptr(client, callback, arg) == 0);
+# endif
+    }
+#endif
+    return false;
+}
+
+bool jackbridge_set_process_thread(jack_client_t* client, JackThreadCallback callback, void* arg)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return (jack_set_process_thread(client, callback, arg) == 0);
+#else
+    if (usingRealJACK && getBridgeInstance().set_process_thread_ptr != nullptr)
+    {
+# ifdef __WINE__
+        WineBridge::getInstance().set_process_thread(callback);
+        return (getBridgeInstance().set_process_thread_ptr(client, WineBridge::process_thread, arg) == 0);
+# else
+        return (getBridgeInstance().set_process_thread_ptr(client, callback, arg) == 0);
+# endif
+    }
+#endif
+    return false;
+}
+
+jack_nframes_t jackbridge_cycle_wait(jack_client_t* client)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    return jack_cycle_wait(client);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().cycle_wait_ptr != nullptr)
+            return getBridgeInstance().cycle_wait_ptr(client);
+#endif
+    return 0;
+}
+
+void jackbridge_cycle_signal(jack_client_t* client, int status)
+{
+#if defined(JACKBRIDGE_DUMMY)
+#elif defined(JACKBRIDGE_DIRECT)
+    jack_cycle_signal(client, status);
+#else
+    if (usingRealJACK)
+        if (getBridgeInstance().cycle_signal_ptr != nullptr)
+            getBridgeInstance().cycle_signal_ptr(client, status);
+#endif
+}
+
+// -----------------------------------------------------------------------------
+
+#ifndef JACKBRIDGE_SKIP_NATIVE_UTILS
+
+START_NAMESPACE_DISTRHO
+
+bool isUsingNativeAudio() noexcept
+{
+#if defined(JACKBRIDGE_DUMMY) || defined(JACKBRIDGE_DIRECT)
+    return false;
+#else
+    return usingNativeBridge;
+#endif
+}
+
+bool supportsAudioInput()
+{
+#if !(defined(JACKBRIDGE_DUMMY) || defined(JACKBRIDGE_DIRECT))
+    if (usingNativeBridge)
+        return nativeBridge->supportsAudioInput();
+#endif
+    return false;
+}
+
+bool supportsBufferSizeChanges()
+{
+#if !(defined(JACKBRIDGE_DUMMY) || defined(JACKBRIDGE_DIRECT))
+    if (usingNativeBridge)
+        return nativeBridge->supportsBufferSizeChanges();
+#endif
+    return false;
+}
+
+bool supportsMIDI()
+{
+#if !(defined(JACKBRIDGE_DUMMY) || defined(JACKBRIDGE_DIRECT))
+    if (usingNativeBridge)
+        return nativeBridge->supportsMIDI();
+#endif
+    return false;
+}
+
+bool isAudioInputEnabled()
+{
+#if !(defined(JACKBRIDGE_DUMMY) || defined(JACKBRIDGE_DIRECT))
+    if (usingNativeBridge)
+        return nativeBridge->isAudioInputEnabled();
+#endif
+    return false;
+}
+
+bool isMIDIEnabled()
+{
+#if !(defined(JACKBRIDGE_DUMMY) || defined(JACKBRIDGE_DIRECT))
+    if (usingNativeBridge)
+        return nativeBridge->isMIDIEnabled();
+#endif
+    return false;
+}
+
+uint getBufferSize()
+{
+#if !(defined(JACKBRIDGE_DUMMY) || defined(JACKBRIDGE_DIRECT))
+    if (usingNativeBridge)
+        return nativeBridge->getBufferSize();
+#endif
+    return 0;
+}
+
+bool requestAudioInput()
+{
+#if !(defined(JACKBRIDGE_DUMMY) || defined(JACKBRIDGE_DIRECT))
+    if (usingNativeBridge)
+        return nativeBridge->requestAudioInput();
+#endif
+    return false;
+}
+
+bool requestBufferSizeChange(const uint newBufferSize)
+{
+#if !(defined(JACKBRIDGE_DUMMY) || defined(JACKBRIDGE_DIRECT))
+    if (usingNativeBridge)
+        return nativeBridge->requestBufferSizeChange(newBufferSize);
+#endif
+    return false;
+}
+
+bool requestMIDI()
+{
+#if !(defined(JACKBRIDGE_DUMMY) || defined(JACKBRIDGE_DIRECT))
+    if (usingNativeBridge)
+        return nativeBridge->requestMIDI();
+#endif
+    return false;
+}
+
+END_NAMESPACE_DISTRHO
+
+#endif // JACKBRIDGE_SKIP_NATIVE_UTILS
+
+// -----------------------------------------------------------------------------