annotate pyvamp-main.cpp @ 0:e20e214bdfb5

Added VAMP-Python binding project vampy
author fazekasgy
date Tue, 11 Mar 2008 19:47:34 +0000
parents
children dc88002ce687
rev   line source
fazekasgy@0 1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
fazekasgy@0 2
fazekasgy@0 3 /*
fazekasgy@0 4 Vamp
fazekasgy@0 5
fazekasgy@0 6 An API for audio analysis and feature extraction plugins.
fazekasgy@0 7
fazekasgy@0 8 Centre for Digital Music, Queen Mary, University of London.
fazekasgy@0 9 Copyright 2006 Chris Cannam.
fazekasgy@0 10
fazekasgy@0 11 Permission is hereby granted, free of charge, to any person
fazekasgy@0 12 obtaining a copy of this software and associated documentation
fazekasgy@0 13 files (the "Software"), to deal in the Software without
fazekasgy@0 14 restriction, including without limitation the rights to use, copy,
fazekasgy@0 15 modify, merge, publish, distribute, sublicense, and/or sell copies
fazekasgy@0 16 of the Software, and to permit persons to whom the Software is
fazekasgy@0 17 furnished to do so, subject to the following conditions:
fazekasgy@0 18
fazekasgy@0 19 The above copyright notice and this permission notice shall be
fazekasgy@0 20 included in all copies or substantial portions of the Software.
fazekasgy@0 21
fazekasgy@0 22 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
fazekasgy@0 23 EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
fazekasgy@0 24 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
fazekasgy@0 25 NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR
fazekasgy@0 26 ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
fazekasgy@0 27 CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
fazekasgy@0 28 WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
fazekasgy@0 29
fazekasgy@0 30 Except as contained in this notice, the names of the Centre for
fazekasgy@0 31 Digital Music; Queen Mary, University of London; and Chris Cannam
fazekasgy@0 32 shall not be used in advertising or otherwise to promote the sale,
fazekasgy@0 33 use or other dealings in this Software without prior written
fazekasgy@0 34 authorization.
fazekasgy@0 35 */
fazekasgy@0 36
fazekasgy@0 37 /**
fazekasgy@0 38 * This VAMP plugin is a wrapper for Python Scripts. (VamPy)
fazekasgy@0 39 * Centre for Digital Music, Queen Mary, University of London.
fazekasgy@0 40 * Copyright 2008, George Fazekas.
fazekasgy@0 41
fazekasgy@0 42 TODO: needs more complete error checking
fazekasgy@0 43 needs correct implementation of Python threading
fazekasgy@0 44 more efficient data conversion using the buffering interface or ctypes
fazekasgy@0 45 VAMP 'programs' not implemented
fazekasgy@0 46 support multiple plugins per script in scanner
fazekasgy@0 47 ensure proper cleanup, (host do a good job though)
fazekasgy@0 48
fazekasgy@0 49 COMPILING AND LINKING:
fazekasgy@0 50 (1) include Python.h wherever it is on your machine
fazekasgy@0 51 (2) this plugin needs to be linked against the Python binary:
fazekasgy@0 52
fazekasgy@0 53 example on on MacOSX:
fazekasgy@0 54 g++ -O2 -Wall -I. -fPIC -c -o pyvamp-main.o pyvamp-main.cpp
fazekasgy@0 55 g++ -dynamiclib -o vamp-pyvamp-plugin.dylib ./PyPlugScanner.o ./PyPlugin.o ./pyvamp-main.o sdk/vamp-sdk/libvamp-sdk.a
fazekasgy@0 56 ... -u _PyMac_Error /Library/Frameworks/Python.framework/Versions/2.5/Python
fazekasgy@0 57 */
fazekasgy@0 58
fazekasgy@0 59 //#include "Python.h"
fazekasgy@0 60 #include "/usr/include/python/Python.h"
fazekasgy@0 61 #include "vamp/vamp.h"
fazekasgy@0 62 #include "vamp-sdk/PluginAdapter.h"
fazekasgy@0 63 #include "PyPlugScanner.h"
fazekasgy@0 64 #include "PyPlugin.h"
fazekasgy@0 65
fazekasgy@0 66 #ifdef _WIN32
fazekasgy@0 67 #define pathsep ('\\')
fazekasgy@0 68 #include <windows.h>
fazekasgy@0 69 #include <tchar.h>
fazekasgy@0 70 #else
fazekasgy@0 71 #define pathsep ('/')
fazekasgy@0 72 #include <dirent.h>
fazekasgy@0 73 #include <dlfcn.h>
fazekasgy@0 74 #endif
fazekasgy@0 75
fazekasgy@0 76 using std::cerr;
fazekasgy@0 77 using std::endl;
fazekasgy@0 78 using std::string;
fazekasgy@0 79 using std::vector;
fazekasgy@0 80
fazekasgy@0 81 volatile bool mutex = false;
fazekasgy@0 82 static int adinstcount;
fazekasgy@0 83
fazekasgy@0 84 class PyPluginAdapter : public Vamp::PluginAdapterBase
fazekasgy@0 85 {
fazekasgy@0 86 public:
fazekasgy@0 87 PyPluginAdapter(std::string pyPlugId, PyObject* pyInstance) :
fazekasgy@0 88 PluginAdapterBase(),
fazekasgy@0 89 m_plug(pyPlugId),
fazekasgy@0 90 m_pyInstance(pyInstance)
fazekasgy@0 91 {
fazekasgy@0 92 cerr << "PyPluginAdapter:ctor:"<< adinstcount << ": " << m_plug << endl;
fazekasgy@0 93 adinstcount++;
fazekasgy@0 94 m_instanceCount = 0;
fazekasgy@0 95 }
fazekasgy@0 96
fazekasgy@0 97 ~PyPluginAdapter()
fazekasgy@0 98 {
fazekasgy@0 99 }
fazekasgy@0 100
fazekasgy@0 101 protected:
fazekasgy@0 102 Vamp::Plugin *createPlugin(float inputSampleRate) {
fazekasgy@0 103
fazekasgy@0 104 std::string pclass = m_plug.substr(m_plug.rfind(':')+1,m_plug.size()-1);
fazekasgy@0 105 std::string ppath = m_plug.substr(0,m_plug.rfind(pathsep));
fazekasgy@0 106 PyPlugin *plugin = new PyPlugin(m_plug,inputSampleRate,m_pyInstance);
fazekasgy@0 107 m_instanceCount++;
fazekasgy@0 108 cerr << "PyPluginAdapter::createPlugin:" << pclass << " (instance: " << m_instanceCount << ")" << endl;
fazekasgy@0 109 return plugin;
fazekasgy@0 110
fazekasgy@0 111 }
fazekasgy@0 112
fazekasgy@0 113 std::string m_plug;
fazekasgy@0 114 bool m_haveInitialized;
fazekasgy@0 115 PyObject *m_pyInstance;
fazekasgy@0 116 int m_instanceCount;
fazekasgy@0 117
fazekasgy@0 118 };
fazekasgy@0 119
fazekasgy@0 120
fazekasgy@0 121 static std::vector<PyPluginAdapter *> adapters;
fazekasgy@0 122 static bool haveScannedPlugins = false;
fazekasgy@0 123
fazekasgy@0 124 const VampPluginDescriptor
fazekasgy@0 125 *vampGetPluginDescriptor(unsigned int version,unsigned int index)
fazekasgy@0 126 {
fazekasgy@0 127 if (version < 1) return 0;
fazekasgy@0 128
fazekasgy@0 129 int isPythonInitialized = Py_IsInitialized();
fazekasgy@0 130 //cerr << "# isPythonInitialized: " << isPythonInitialized << endl;
fazekasgy@0 131 //cerr << "# haveScannedPlugins: " << haveScannedPlugins << endl;
fazekasgy@0 132
fazekasgy@0 133 if (!haveScannedPlugins) {
fazekasgy@0 134
fazekasgy@0 135 if (!isPythonInitialized) {
fazekasgy@0 136
fazekasgy@0 137 string pythonPath =
fazekasgy@0 138 (string) Py_GetExecPrefix() + pathsep +
fazekasgy@0 139 (string) Py_GetProgramName();
fazekasgy@0 140
fazekasgy@0 141 void *pylib = 0;
fazekasgy@0 142
fazekasgy@0 143 cerr << "Loading Python Interpreter at: " << pythonPath << endl;
fazekasgy@0 144 //Preloading the binary allows the load of shared libs //dlopen("/Library/Frameworks/Python.framework/Versions/2.5/Python", RTLD_NOW|RTLD_GLOBAL);
fazekasgy@0 145 #ifdef _WIN32
fazekasgy@0 146 //TODO: check how to do RTLD_NOW on Windows
fazekasgy@0 147 pylib = LoadLibrary(pythonPath.c_str());
fazekasgy@0 148 #else
fazekasgy@0 149 pylib = dlopen(pythonPath.c_str(), RTLD_NOW|RTLD_GLOBAL);
fazekasgy@0 150 #endif
fazekasgy@0 151 if (!pylib) cerr << "Warning: Could not preload Python."
fazekasgy@0 152 << " Dynamic lodading in scripts will fail." << endl;
fazekasgy@0 153 Py_Initialize();
fazekasgy@0 154 PyEval_InitThreads();
fazekasgy@0 155 } else {
fazekasgy@0 156 //Py_InitializeEx(1);
fazekasgy@0 157 }
fazekasgy@0 158
fazekasgy@0 159 vector<string> pyPlugs;
fazekasgy@0 160 vector<string> pyPath;
fazekasgy@0 161 vector<PyObject *> pyInstances;
fazekasgy@0 162 static PyPlugScanner *scanner;
fazekasgy@0 163
fazekasgy@0 164 //Scanning Plugins
fazekasgy@0 165 cerr << "Scanning PyPlugins" << endl;
fazekasgy@0 166 scanner = PyPlugScanner::getInstance();
fazekasgy@0 167 pyPath=scanner->getAllValidPath();
fazekasgy@0 168 //add this as extra path for development
fazekasgy@0 169 pyPath.push_back("/Users/Shared/Development/vamp-experiments");
fazekasgy@0 170 scanner->setPath(pyPath);
fazekasgy@0 171 pyPlugs = scanner->getPyPlugs();
fazekasgy@0 172 cerr << "Found " << pyPlugs.size() << " Scripts ...OK" << endl;
fazekasgy@0 173 //TODO: this will support multiple classes per script
fazekasgy@0 174 pyInstances = scanner->getPyInstances();
fazekasgy@0 175 cerr << "Found " << pyInstances.size() << " Instances ...OK" << endl;
fazekasgy@0 176
fazekasgy@0 177 for (size_t i = 0; i < pyPlugs.size(); ++i) {
fazekasgy@0 178 adapters.push_back( new PyPluginAdapter(pyPlugs[i],pyInstances[i]));
fazekasgy@0 179 }
fazekasgy@0 180 haveScannedPlugins=true;
fazekasgy@0 181 }
fazekasgy@0 182
fazekasgy@0 183 cerr << "Accessing adapter index: " << index << " (adapters: " << adapters.size() << ")" << endl;
fazekasgy@0 184 if (index<adapters.size()) {
fazekasgy@0 185 const VampPluginDescriptor *tmp = adapters[index]->getDescriptor();
fazekasgy@0 186 return tmp;
fazekasgy@0 187 } else return 0;
fazekasgy@0 188
fazekasgy@0 189
fazekasgy@0 190 }
fazekasgy@0 191
fazekasgy@0 192
fazekasgy@0 193
fazekasgy@0 194
fazekasgy@0 195
fazekasgy@0 196
fazekasgy@0 197
fazekasgy@0 198