/* * Copyright (C) 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 #ifdef PLATFORM_WINDOWS #include #else #include #endif #ifdef COMPILER_MSVC #include #else #include #include #endif #ifdef __MINGW64__ #define NO_OLDNAMES // no backwards compat _pid_t, conflict with w64 pthread/sched #endif #include "pbd/error.h" #include "pbd/transmitter.h" #include "pbd/receiver.h" #include "pbd/pbd.h" #include "pbd/stacktrace.h" #include "pbd/win_console.h" #include "pbd/xml++.h" #include "ardour/types.h" #include "ardour/vst_types.h" #ifdef __MINGW64__ #undef NO_OLDNAMES #endif #include "../ardour/filesystem_paths.cc" #include "../ardour/vst_state.cc" #include "../ardour/vst2_scan.cc" #ifdef LXVST_SUPPORT #include "../ardour/linux_vst_support.cc" void vstfx_destroy_editor (VSTState* /*vstfx*/) { } #endif #ifdef MACVST_SUPPORT #include "../ardour/mac_vst_support.cc" #endif using namespace PBD; class LogReceiver : public Receiver { protected: void receive (Transmitter::Channel chn, const char * str) { const char *prefix = ""; switch (chn) { case Transmitter::Debug: /* ignore */ break; case Transmitter::Info: prefix = "[Info]: "; break; case Transmitter::Warning: prefix = "[WARNING]: "; break; case Transmitter::Error: prefix = "[ERROR]: "; break; case Transmitter::Fatal: prefix = "[FATAL]: "; break; case Transmitter::Throw: abort (); } std::cout << prefix << str << std::endl; if (chn == Transmitter::Fatal) { console_madness_end (); ::exit (EXIT_FAILURE); } } }; LogReceiver log_receiver; static void vst2_plugin (std::string const& module_path, PluginType, VST2Info const& i) { info << "Found Plugin: '" << ARDOUR::vst2_id_to_str (i.id) << "' " << i.name << endmsg; } static bool scan_vst2 (std::string const& path, ARDOUR::PluginType type, bool force, bool verbose) { info << "Scanning: " << path << endmsg; if (!vst2_valid_cache_file (path, verbose).empty()) { if (!force) { info << "Skipping scan." << endmsg; return true; } } if (vst2_scan_and_cache (path, type, sigc::ptr_fun (&vst2_plugin), verbose)) { info << string_compose (_("Saved VST2 plugin cache to %1"), vst2_cache_file (path)) << endmsg; } return true; } #ifdef PLATFORM_WINDOWS static LONG WINAPI crash_handler (EXCEPTION_POINTERS* exceptioninfo) { // TODO consider DrMingw if HAVE_DRMINGW printf ("Error: %x\n ---8<---\n", exceptioninfo->ExceptionRecord->ExceptionCode); PBD::stacktrace (std::cout, 15, 2); printf (" --->8---\n"); return EXCEPTION_CONTINUE_SEARCH; } #else static void sig_handler (int sig) { printf ("Error: signal %d\n ---8<---\n", sig); PBD::stacktrace (std::cout, 15, 2); printf (" --->8---\n"); fflush(stdout); fflush(stderr); _exit (EXIT_FAILURE); } #endif static void usage () { // help2man compatible format (standard GNU help-text) printf ("ardour-vst2-scanner - load and index VST2 plugins.\n\n"); printf ("Usage: ardour-vst2-scanner [ OPTIONS ] [VST2-file]*\n\n"); printf ("Options:\n\ -f, --force Force update of cache file\n\ -h, --help Display this help and exit\n\ -q, --quiet Hide usual output, only print errors\n\ -v, --verbose Give verbose output (unless quiet)\n\ -V, --version Print version information and exit\n\ \n"); printf ("\n\ This tool ...\n\ \n"); printf ("Report bugs to \n" "Website: \n"); console_madness_end (); ::exit (EXIT_SUCCESS); } int main (int argc, char **argv) { bool print_log = true; bool stop_on_error = false; bool force = false; bool verbose = false; const char* optstring = "fhqvV"; /* clang-format off */ const struct option longopts[] = { { "force", no_argument, 0, 'f' }, { "help", no_argument, 0, 'h' }, { "quiet", no_argument, 0, 'q' }, { "verbose", no_argument, 0, 'v' }, { "version", no_argument, 0, 'V' }, }; /* clang-format on */ console_madness_begin (); int c = 0; while (EOF != (c = getopt_long (argc, argv, optstring, longopts, (int*)0))) { switch (c) { case 'V': printf ("ardour-vst2-scanner version %s\n\n", VERSIONSTRING); printf ("Copyright (C) GPL 2021 Robin Gareus \n"); console_madness_end (); exit (EXIT_SUCCESS); break; case 'f': force = true; break; case 'h': usage (); break; case 'q': print_log = false; break; case 'v': verbose = true; break; default: std::cerr << "Error: unrecognized option. See --help for usage information.\n"; console_madness_end (); ::exit (EXIT_FAILURE); break; } } if (optind >= argc) { std::cerr << "Error: Missing parameter. See --help for usage information.\n"; console_madness_end (); ::exit (EXIT_FAILURE); } PBD::init(); if (print_log) { log_receiver.listen_to (info); log_receiver.listen_to (warning); log_receiver.listen_to (error); log_receiver.listen_to (fatal); } else { verbose = false; } bool err = false; #ifdef PLATFORM_WINDOWS ::SetUnhandledExceptionFilter (crash_handler); #else signal (SIGSEGV, sig_handler); signal (SIGBUS, sig_handler); signal (SIGILL, sig_handler); signal (SIGABRT, sig_handler); #endif while (optind < argc) { const char* dllpath = argv[optind++]; const size_t slen = strlen (dllpath); ARDOUR::PluginType type; if (false) { } #ifdef LXVST_SUPPORT else if (slen > 3 && 0 == g_ascii_strcasecmp (&dllpath[slen-3], ".so")) { type = ARDOUR::LXVST; } #endif #ifdef WINDOWS_VST_SUPPORT else if (slen > 4 && 0 == g_ascii_strcasecmp (&dllpath[slen-4], ".dll")) { type = ARDOUR::Windows_VST; } #endif #ifdef MACVST_SUPPORT else if (slen > 4 && 0 == g_ascii_strcasecmp (&dllpath[slen-4], ".vst")) { type = ARDOUR::MacVST; } #endif else { error << "'" << dllpath << "' is not a supported VST plugin." << endmsg; //fprintf(stderr, "'%s' is not a supported VST plugin.\n", dllpath); continue; } if (!scan_vst2 (dllpath, type, force, verbose)) { err = true; } if (stop_on_error) { break; } } PBD::cleanup(); console_madness_end (); if (err) { return EXIT_FAILURE; } else { return EXIT_SUCCESS; } }