271 lines
6.2 KiB
C++
271 lines
6.2 KiB
C++
/*
|
|
* Copyright (C) 2021 Robin Gareus <robin@gareus.org>
|
|
*
|
|
* 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 <cstdlib>
|
|
#include <cstdio>
|
|
#include <ctype.h>
|
|
#include <getopt.h>
|
|
#include <iostream>
|
|
#include <string>
|
|
|
|
#ifdef COMPILER_MSVC
|
|
#include <sys/utime.h>
|
|
#else
|
|
#include <utime.h>
|
|
#include <strings.h>
|
|
#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/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;
|
|
}
|
|
|
|
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> [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 <http://tracker.ardour.org/>\n"
|
|
"Website: <http://ardour.org/>\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 <robin@gareus.org>\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;
|
|
|
|
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;
|
|
}
|
|
}
|