/* * Copyright (C) 2016-2021 Robin Gareus * * 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 * (at your option) 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. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include #include #include #include #include #include #include #include #include #include #include #include #include "pbd/gstdio_compat.h" #include #include #include "ardour/mac_vst_support.h" #include "pbd/basename.h" #include "pbd/error.h" #include "pbd/i18n.h" /*Simple error handler stuff for VSTFX*/ void mac_vst_error (const char *fmt, ...) { va_list ap; char buffer[512]; va_start (ap, fmt); vsnprintf (buffer, sizeof (buffer), fmt, ap); mac_vst_error_callback (buffer); va_end (ap); } /*default error handler callback*/ static void default_mac_vst_error_callback (const char *desc) { PBD::error << desc << endmsg; } void (*mac_vst_error_callback)(const char *desc) = &default_mac_vst_error_callback; /* --- */ /*Create and return a pointer to a new VSTFX handle*/ static VSTHandle * mac_vst_handle_new () { VSTHandle* mac_vst = (VSTHandle *) calloc (1, sizeof (VSTHandle)); return mac_vst; } /*Create and return a pointer to a new mac_vst instance*/ static VSTState * mac_vst_new () { VSTState* mac_vst = (VSTState *) calloc (1, sizeof (VSTState)); vststate_init (mac_vst); return mac_vst; } /*This loads up a plugin, given the path to its .vst bundle and * finds its main entry point etc */ VSTHandle * mac_vst_load (const char *path) { VSTHandle* fhandle; /*Create a new handle we can use to reference the plugin*/ fhandle = mac_vst_handle_new (); fhandle->bundleRef = 0; CFURLRef url; if (!(url = CFURLCreateFromFileSystemRepresentation (0, (const UInt8*)path, (CFIndex) strlen (path), true))) { return 0; } fhandle->bundleRef = CFBundleCreate (kCFAllocatorDefault, url); CFRelease (url); if (fhandle->bundleRef == 0) { free (fhandle); return 0; } if (!CFBundleLoadExecutable (fhandle->bundleRef)) { free (fhandle); return 0; } fhandle->name = strdup (path); fhandle->main_entry = (main_entry_t) CFBundleGetFunctionPointerForName (fhandle->bundleRef, CFSTR("main_macho")); if (!fhandle->main_entry) { fhandle->main_entry = (main_entry_t) CFBundleGetFunctionPointerForName (fhandle->bundleRef, CFSTR("VSTPluginMain")); } if (fhandle->main_entry == 0) { PBD::error << string_compose (_("Missing entry method in VST2 plugin '%1'"), path) << endmsg; mac_vst_unload (fhandle); return 0; } fhandle->res_file_id = CFBundleOpenBundleResourceMap (fhandle->bundleRef); /*return the handle of the plugin*/ return fhandle; } /*This unloads a plugin*/ int mac_vst_unload (VSTHandle* fhandle) { if (fhandle->plugincnt) { /*Still have plugin instances - can't unload the library */ return -1; } /*Valid plugin loaded?*/ if (fhandle->bundleRef) { CFBundleCloseBundleResourceMap (fhandle->bundleRef, fhandle->res_file_id); fhandle->bundleRef = 0; } if (fhandle->name) { free (fhandle->name); fhandle->name = 0; } /*Don't need the plugin handle any more*/ free (fhandle); return 0; } /*This instantiates a plugin*/ VSTState * mac_vst_instantiate (VSTHandle* fhandle, audioMasterCallback amc, void* userptr) { VSTState* mac_vst = mac_vst_new (); if (fhandle == 0) { mac_vst_error ( "** ERROR ** VSTFX : The handle was 0\n" ); free (mac_vst); return 0; } if ((mac_vst->plugin = fhandle->main_entry (amc)) == 0) { mac_vst_error ("** ERROR ** VSTFX : %s could not be instantiated :(\n", fhandle->name); free (mac_vst); return 0; } mac_vst->handle = fhandle; mac_vst->plugin->ptr1 = userptr; if (mac_vst->plugin->magic != kEffectMagic) { mac_vst_error ("** ERROR ** VSTFX : %s is not a VST plugin\n", fhandle->name); free (mac_vst); return 0; } if (!userptr) { /* scanning.. or w/o master-callback userptr == 0, open now. * * Session::vst_callback needs a pointer to the AEffect * ((VSTPlugin*)userptr)->_plugin = vstfx->plugin * before calling effOpen, because effOpen may call back */ mac_vst->plugin->dispatcher (mac_vst->plugin, effOpen, 0, 0, 0, 0); mac_vst->vst_version = mac_vst->plugin->dispatcher (mac_vst->plugin, effGetVstVersion, 0, 0, 0, 0); /* configure plugin to use Cocoa View */ mac_vst->plugin->dispatcher (mac_vst->plugin, effCanDo, 0, 0, const_cast ("hasCockosViewAsConfig"), 0.0f); } mac_vst->handle->plugincnt++; mac_vst->wantIdle = 0; return mac_vst; } /*Close a mac_vst instance*/ void mac_vst_close (VSTState* mac_vst) { // assert that the GUI object is destoyed if (mac_vst->plugin) { mac_vst->plugin->dispatcher (mac_vst->plugin, effMainsChanged, 0, 0, 0, 0); /*Calling dispatcher with effClose will cause the plugin's destructor to be called, which will also remove the editor if it exists*/ mac_vst->plugin->dispatcher (mac_vst->plugin, effClose, 0, 0, 0, 0); } if (mac_vst->handle->plugincnt) { mac_vst->handle->plugincnt--; } /* mac_vst_unload will unload the dll if the instance count allows - * we need to do this because some plugins keep their own instance count * and (JUCE) manages the plugin UI in its own thread. When the plugins * internal instance count reaches zero, JUCE stops the UI thread and won't * restart it until the next time the library is loaded. If we don't unload * the lib JUCE will never restart */ mac_vst_unload (mac_vst->handle); free (mac_vst); } #if 0 // TODO wrap dispatch intptr_t mac_vst_dispatch (VSTState* mac_vst, int op, int idx, intptr_t val, void* ptr, float opt) { const ResFileRefNum old_resources = CurResFile(); if (mac_vst->handle->res_file_id) { UseResFile (mac_vst->handle->res_file_id); } mac_vst->plugin->dispatcher (mac_vst->plugin, op, idx, val, prt, opt); const ResFileRefNum current_res = CurResFile(); if (current_res != old_resources) { mac_vst->handle->res_file_id = current_res; UseResFile (old_resources); } } #endif