Chris@12: /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */ Chris@12: Chris@12: #include Chris@12: #include Chris@12: #include Chris@12: #include Chris@12: #include Chris@12: Chris@12: #include Chris@12: #include Chris@12: #include Chris@12: Chris@12: #include Chris@12: #include Chris@12: #include Chris@12: Chris@12: #include Chris@12: #include Chris@12: Chris@12: using std::cout; Chris@12: using std::cin; Chris@12: using std::cerr; Chris@12: using std::getline; Chris@12: using std::endl; Chris@12: using std::string; Chris@12: using std::vector; Chris@12: using std::ofstream; Chris@12: using std::ios; Chris@12: Chris@12: using Vamp::HostExt::PluginLoader; Chris@12: using Vamp::Plugin; Chris@12: Chris@12: //??? Chris@12: string programURI = "http://www.vamp-plugins.org/doap.rdf#template-generator"; Chris@12: Chris@12: void usage() Chris@12: { Chris@12: cerr << endl; Chris@12: cerr << "vamp-rdf-template-generator: Create a skeleton RDF description file describing" << endl; Chris@12: cerr << "a Vamp plugin library using the Vamp ontology." << endl; Chris@12: cerr << endl; Chris@12: cerr << "Usage:" << endl; Chris@12: cerr << " vamp-rdf-template-generator -i vamp:soname[:plugin] [vamp:soname[:plugin] ...]" << endl; Chris@12: cerr << " vamp-rdf-template-generator PLUGIN_BASE_URI [ -m YOUR_URI ] [vamp:]soname[:plugin] [[vamp:]soname[:plugin] ...]" << endl; Chris@12: cerr << endl; Chris@12: cerr << "Example:" << endl; Chris@12: cerr << " vamp-rdf-template-generator http://vamp-plugins.org/rdf/plugins/ vamp-example-plugins" << endl; Chris@12: cerr << endl; Chris@12: exit(2); Chris@12: } Chris@12: Chris@12: template Chris@12: inline string to_string (const T& t) Chris@12: { Chris@12: std::stringstream ss; Chris@12: ss << t; Chris@12: return ss.str(); Chris@12: } Chris@12: Chris@12: string describe_namespaces(string pluginBundleBaseURI, string libname) Chris@12: { Chris@12: string res=\ Chris@12: "@prefix rdfs: .\n\ Chris@12: @prefix xsd: .\n\ Chris@12: @prefix vamp: .\n\ Chris@12: @prefix plugbase: <"+pluginBundleBaseURI+libname+"#> .\n\ Chris@12: @prefix owl: .\n\ Chris@12: @prefix dc: .\n\ Chris@12: @prefix af: .\n\ Chris@12: @prefix foaf: .\n\ Chris@12: @prefix cc: .\n\ Chris@12: @prefix : <#> .\n\n"; Chris@12: Chris@12: return res; Chris@12: } Chris@12: Chris@12: string describe_doc(string describerURI, string pluginBundleBaseURI, Chris@12: string libname) Chris@12: { Chris@12: string res=\ Chris@12: "<> a vamp:PluginDescription ;\n"; Chris@12: if (describerURI != "") { Chris@12: res += " foaf:maker <"+describerURI+"> ;\n"; Chris@12: } Chris@12: res += "\ Chris@12: foaf:maker <"+programURI+"> ;\n\ Chris@12: foaf:primaryTopic <"+pluginBundleBaseURI+libname+"> .\n\n"; Chris@12: return res; Chris@12: } Chris@12: Chris@12: Chris@12: string describe_library(string libname, vector plugins) Chris@12: { Chris@12: string res=\ Chris@12: ":"+libname+" a vamp:PluginLibrary ;\n\ Chris@12: vamp:identifier \""+libname+"\" "; Chris@12: Chris@12: for (size_t i = 0; i < plugins.size(); ++i) { Chris@12: res += " ; \n\ Chris@12: vamp:available_plugin plugbase:"+plugins[i]->getIdentifier(); Chris@12: } Chris@12: Chris@12: res += " ; \n\ Chris@12: # foaf:page ;\n\ Chris@12: .\n\n"; Chris@12: return res; Chris@12: } Chris@12: Chris@12: string describe_plugin(Plugin* plugin) Chris@12: { Chris@12: string res=\ Chris@12: "plugbase:"+plugin->getIdentifier()+" a vamp:Plugin ;\n\ Chris@12: dc:title \""+plugin->getName()+"\" ;\n\ Chris@12: vamp:name \""+plugin->getName()+"\" ;\n\ Chris@12: dc:description \"\"\""+plugin->getDescription()+"\"\"\" ;\n\ Chris@12: foaf:maker [ foaf:name \""+plugin->getMaker()+"\" ] ; # FIXME could give plugin author's URI here\n\ Chris@12: dc:rights \"\"\""+plugin->getCopyright()+"\"\"\" ;\n\ Chris@12: # cc:license ; \n\ Chris@12: vamp:identifier \""+plugin->getIdentifier()+"\" ;\n\ Chris@12: vamp:vamp_API_version vamp:api_version_"+to_string(plugin->getVampApiVersion())+" ;\n\ Chris@12: owl:versionInfo \""+to_string(plugin->getPluginVersion())+"\" ;\n"; Chris@12: if (plugin->getInputDomain() == Vamp::Plugin::FrequencyDomain) Chris@12: res+=" vamp:input_domain vamp:FrequencyDomain ;\n\n"; Chris@12: else Chris@12: res+=" vamp:input_domain vamp:TimeDomain ;\n"; Chris@12: Chris@12: Chris@12: Plugin::ParameterList params = plugin->getParameterDescriptors(); Chris@12: if (!params.empty()) res+="\n"; Chris@12: for (Plugin::ParameterList::const_iterator i = params.begin(); i != params.end(); i++) Chris@12: res+=" vamp:parameter plugbase:"+plugin->getIdentifier()+"_param_"+(*i).identifier+" ;\n"; Chris@12: if (!params.empty()) res+="\n"; Chris@12: Chris@12: Plugin::OutputList outputs = plugin->getOutputDescriptors(); Chris@12: for (Plugin::OutputList::const_iterator i = outputs.begin(); i!= outputs.end(); i++) Chris@12: res+=" vamp:output plugbase:"+plugin->getIdentifier()+"_output_"+(*i).identifier+" ;\n"; Chris@12: res+=" .\n"; Chris@12: Chris@12: return res; Chris@12: } Chris@12: Chris@12: string describe_param(Plugin *plugin, Plugin::ParameterDescriptor p) Chris@12: { Chris@12: Chris@12: //FIXME: dc:format and vamp:unit are the same??? Chris@12: //Should be a QUantizedParameter also a Parameter?? Chris@12: if(p.isQuantized){ Chris@12: string res=\ Chris@12: "plugbase:"+plugin->getIdentifier()+"_param_"+p.identifier+" a vamp:QuantizedParameter ;\n\ Chris@12: vamp:identifier \""+p.identifier+"\" ;\n\ Chris@12: dc:title \""+p.name+"\" ;\n\ Chris@12: dc:format \""+p.unit+"\" ;\n\ Chris@12: vamp:min_value "+to_string(p.minValue)+" ;\n\ Chris@12: vamp:max_value "+to_string(p.maxValue)+" ;\n\ Chris@12: vamp:unit \""+p.unit+"\" ;\n\ Chris@12: vamp:quantize_step "+to_string(p.quantizeStep)+" ;\n\ Chris@12: vamp:default_value "+to_string(p.defaultValue)+" ;\n\ Chris@12: vamp:value_names ("; Chris@12: Chris@12: unsigned int i; Chris@12: for (i=0; i+1 < p.valueNames.size(); i++) Chris@12: res+=" \""+p.valueNames[i]+"\""; Chris@12: if (i < p.valueNames.size()) Chris@12: res+=" \""+p.valueNames[i]+"\""; Chris@12: res+=");\n"; Chris@12: Chris@12: res+=" .\n"; Chris@12: Chris@12: return res; Chris@12: Chris@12: }else{ Chris@12: string res=\ Chris@12: "plugbase:"+plugin->getIdentifier()+"_param_"+p.identifier+" a vamp:Parameter ;\n\ Chris@12: vamp:identifier \""+p.identifier+"\" ;\n\ Chris@12: dc:title \""+p.name+"\" ;\n\ Chris@12: dc:format \""+p.unit+"\" ;\n\ Chris@12: vamp:min_value "+to_string(p.minValue)+" ;\n\ Chris@12: vamp:max_value "+to_string(p.maxValue)+" ;\n\ Chris@12: vamp:unit \""+p.unit+"\" ;\n\ Chris@12: vamp:default_value "+to_string(p.defaultValue)+" ;\n\ Chris@12: vamp:value_names ("; Chris@12: Chris@12: unsigned int i; Chris@12: for (i=0; i+1 < p.valueNames.size(); i++) Chris@12: res+=" \""+p.valueNames[i]+"\""; Chris@12: if (i < p.valueNames.size()) Chris@12: res+=" \""+p.valueNames[i]+"\""; Chris@12: res+=");\n"; Chris@12: Chris@12: res+=" .\n"; Chris@12: Chris@12: return res; Chris@12: Chris@12: } Chris@12: } Chris@12: Chris@12: string describe_output(Plugin *plugin, Plugin::OutputDescriptor o) Chris@12: { Chris@12: Chris@12: //we need to distinguish here between different output types: Chris@12: Chris@12: //Quantize or not Chris@12: //KnownExtents or not Chris@12: //Data output classification: Chris@12: //DenseOutput Chris@12: //SparseOutput Chris@12: //TrackLevelOutput Chris@12: Chris@12: Chris@12: // SparseOutput: variable sample rate. Events are not evenly Chris@12: // spaced so we need to record the time associated with the event Chris@12: // as it its not ensured that we have an event after the next one Chris@12: // (but there is not time to set the duration, it has to be Chris@12: // calculated as the different between 2 different events). The Chris@12: // timestamp must be read. Chris@12: Chris@12: string res; Chris@12: Chris@12: if (o.sampleType == Plugin::OutputDescriptor::VariableSampleRate || Chris@12: !o.hasFixedBinCount) Chris@12: { Chris@12: Chris@12: res=\ Chris@12: "plugbase:"+plugin->getIdentifier()+"_output_"+o.identifier+" a vamp:SparseOutput ;\n\ Chris@12: vamp:identifier \""+o.identifier+"\" ;\n\ Chris@12: dc:title \""+o.name+"\" ;\n\ Chris@12: dc:description \"\"\""+o.description+"\"\"\" ;\n\ Chris@12: vamp:fixed_bin_count \""+(o.hasFixedBinCount == 1 ? "true" : "false")+"\" ;\n\ Chris@12: vamp:unit \""+(o.unit)+"\" ;\n"; Chris@12: Chris@12: Chris@12: //another type of output Chris@12: if(o.isQuantized){ Chris@12: Chris@12: res+=" a vamp:QuantizedOutput ;\n"; Chris@12: res+=" vamp:quantize_step "+to_string(o.quantizeStep)+" ;\n"; Chris@12: } Chris@12: Chris@12: //and yet another type Chris@12: if(o.hasKnownExtents){ Chris@12: Chris@12: res+=" a vamp:KnownExtentsOutput ;\n"; Chris@12: res+=" vamp:min_value "+to_string(o.minValue)+" ;\n"; Chris@12: res+=" vamp:max_value "+to_string(o.maxValue)+" ;\n"; Chris@12: } Chris@12: Chris@12: // FIXME ? Bin names may vary based on plugin setup, so including them here might be misleading... Chris@12: if (o.hasFixedBinCount) Chris@12: { Chris@12: res+=" vamp:bin_count "+to_string(o.binCount)+" ;\n"; Chris@12: Chris@12: bool haveBinNames = false; Chris@12: for (unsigned int i=0; i < o.binNames.size(); i++) { Chris@12: if (o.binNames[i] != "") { Chris@12: haveBinNames = true; Chris@12: break; Chris@12: } Chris@12: } Chris@12: Chris@12: if (haveBinNames) { Chris@12: res+=" vamp:bin_names ("; Chris@12: Chris@12: unsigned int i; Chris@12: for (i=0; i+1 < o.binNames.size(); i++) Chris@12: res+=" \""+o.binNames[i]+"\""; Chris@12: if (i < o.binNames.size()) Chris@12: res+=" \""+o.binNames[i]+"\""; Chris@12: res+=");\n"; Chris@12: } Chris@12: } Chris@12: Chris@12: res+=" vamp:sample_type vamp:VariableSampleRate ;\n"; Chris@12: if (o.sampleRate > 0.0f) Chris@12: res+=" vamp:sample_rate "+to_string(o.sampleRate)+" ;\n"; Chris@12: Chris@12: } Chris@12: Chris@12: //If we do not have SparseOutput, then we have DenseOutput. TrackLevelOutput can not be inferred from the plugin directly without actually Chris@12: //running the plugin. Chris@12: else{ Chris@12: Chris@12: res=\ Chris@12: "plugbase:"+plugin->getIdentifier()+"_output_"+o.identifier+" a vamp:DenseOutput ;\n\ Chris@12: vamp:identifier \""+o.identifier+"\" ;\n\ Chris@12: dc:title \""+o.name+"\" ;\n\ Chris@12: dc:description \"\"\""+o.description+"\"\"\" ;\n\ Chris@12: vamp:fixed_bin_count \""+(o.hasFixedBinCount == 1 ? "true" : "false")+"\" ;\n\ Chris@12: vamp:unit \""+(o.unit)+"\" ;\n"; Chris@12: Chris@12: Chris@12: //another type of output Chris@12: if(o.isQuantized){ Chris@12: Chris@12: res+=" a vamp:QuantizedOutput ;\n"; Chris@12: res+=" vamp:quantize_step "+to_string(o.quantizeStep)+" ;\n"; Chris@12: } Chris@12: Chris@12: //and yet another type Chris@12: if(o.hasKnownExtents){ Chris@12: Chris@12: res+=" a vamp:KnownExtentsOutput ;\n"; Chris@12: res+=" vamp:min_value "+to_string(o.minValue)+" ;\n"; Chris@12: res+=" vamp:max_value "+to_string(o.maxValue)+" ;\n"; Chris@12: } Chris@12: Chris@12: // FIXME ? Bin names may vary based on plugin setup, so including them here might be misleading... Chris@12: if (o.hasFixedBinCount) Chris@12: { Chris@12: res+=" vamp:bin_count "+to_string(o.binCount)+" ;\n"; Chris@12: Chris@12: bool haveBinNames = false; Chris@12: for (unsigned int i=0; i < o.binNames.size(); i++) { Chris@12: if (o.binNames[i] != "") { Chris@12: haveBinNames = true; Chris@12: break; Chris@12: } Chris@12: } Chris@12: Chris@12: if (haveBinNames) { Chris@12: res+=" vamp:bin_names ("; Chris@12: Chris@12: unsigned int i; Chris@12: for (i=0; i+1 < o.binNames.size(); i++) Chris@12: res+=" \""+o.binNames[i]+"\""; Chris@12: if (i < o.binNames.size()) Chris@12: res+=" \""+o.binNames[i]+"\""; Chris@12: res+=");\n"; Chris@12: } Chris@12: } Chris@12: Chris@12: else if (o.sampleType == Plugin::OutputDescriptor::FixedSampleRate) Chris@12: { Chris@12: res+=" vamp:sample_type vamp:FixedSampleRate ;\n"; Chris@12: res+=" vamp:sample_rate "+to_string(o.sampleRate)+" ;\n"; Chris@12: } Chris@12: else if (o.sampleType == Plugin::OutputDescriptor::OneSamplePerStep) Chris@12: res+=" vamp:sample_type vamp:OneSamplePerStep ;\n"; Chris@12: else Chris@12: { Chris@12: cerr<<"Incomprehensible sampleType for output descriptor "+o.identifier<<" !"< plugins, string pluginBundleBaseURI, Chris@12: string describerURI, string libname) Chris@12: { Chris@12: string res = describe_namespaces(pluginBundleBaseURI, libname); Chris@12: Chris@12: res += describe_doc(describerURI, pluginBundleBaseURI, libname); Chris@12: Chris@12: res += describe_library(libname, plugins); Chris@12: Chris@12: for (size_t i = 0; i < plugins.size(); ++i) { Chris@12: Chris@12: Plugin *plugin = plugins[i]; Chris@12: Chris@12: res += describe_plugin(plugin); Chris@12: Chris@12: Plugin::ParameterList params = plugin->getParameterDescriptors(); Chris@12: for (Plugin::ParameterList::const_iterator i = params.begin(); i != params.end(); i++) Chris@12: res += describe_param(plugin, *i); Chris@12: Chris@12: Plugin::OutputList outputs = plugin->getOutputDescriptors(); Chris@12: for (Plugin::OutputList::const_iterator i = outputs.begin(); i!= outputs.end(); i++) Chris@12: res += describe_output(plugin, *i); Chris@12: } Chris@12: Chris@12: return res; Chris@12: } Chris@12: Chris@12: int main(int argc, char **argv) Chris@12: { Chris@12: if (argc < 3) usage(); Chris@12: Chris@12: bool interactive = false; Chris@12: if (!strcmp(argv[1], "-i")) interactive = true; Chris@12: Chris@12: if (!interactive && argc < 3) usage(); Chris@12: Chris@12: string pluginBundleBaseURI, describerURI; Chris@12: Chris@12: int argidx = 2; Chris@12: Chris@12: if (!interactive) { Chris@12: pluginBundleBaseURI = argv[1]; Chris@12: if (!strcmp(argv[2], "-m")) { Chris@12: if (argc < 5) usage(); Chris@12: describerURI = argv[3]; Chris@12: argidx = 4; Chris@12: } Chris@12: } else { Chris@12: cerr << "Please enter the base URI for the plugin bundle : "; Chris@12: getline(cin, pluginBundleBaseURI); Chris@12: cerr << "Please enter your URI (empty to omit) : "; Chris@12: getline(cin, describerURI); Chris@12: } Chris@12: Chris@12: vector plugins; Chris@12: string libname; Chris@12: Chris@12: PluginLoader *loader = PluginLoader::getInstance(); Chris@12: Chris@12: while (argidx < argc) { Chris@12: Chris@12: string pluginName = argv[argidx]; Chris@12: Chris@12: if (pluginName.substr(0, 5) == "vamp:") { Chris@12: pluginName = pluginName.substr(5); Chris@12: } Chris@12: Chris@12: string mylibname = pluginName.substr(0, pluginName.find(':')); Chris@12: Chris@12: if (libname == "") libname = mylibname; Chris@12: else if (libname != mylibname) { Chris@12: cerr << "ERROR: All plugins specified on command line must originate in the same library" << endl; Chris@12: exit(1); Chris@12: } Chris@12: Chris@12: if (mylibname == pluginName) { // pluginName is a library, not a plugin Chris@12: Chris@12: PluginLoader::PluginKeyList list = loader->listPlugins(); Chris@12: for (size_t i = 0; i < list.size(); ++i) { Chris@12: string thislibname = list[i].substr(0, list[i].find(':')); Chris@12: if (thislibname != mylibname) continue; Chris@12: Plugin *plugin = loader->loadPlugin(list[i], 44100); Chris@12: if (!plugin) { Chris@12: cerr << "ERROR: Plugin \"" << list[i] << "\" could not be loaded" << endl; Chris@12: exit(1); Chris@12: } Chris@12: plugins.push_back(plugin); Chris@12: } Chris@12: Chris@12: if (plugins.empty()) { Chris@12: cerr << "ERROR: Plugin library \"" << mylibname << "\" does not exist, could not be opened, or contains no plugins" << endl; Chris@12: exit(1); Chris@12: } Chris@12: Chris@12: } else { // pluginName is a plugin Chris@12: Chris@12: Plugin *plugin = loader->loadPlugin(pluginName, size_t(44100)); Chris@12: if (!plugin) { Chris@12: cerr << "ERROR: Plugin \"" << pluginName << "\" could not be loaded" << endl; Chris@12: exit(1); Chris@12: } Chris@12: plugins.push_back(plugin); Chris@12: } Chris@12: Chris@12: ++argidx; Chris@12: } Chris@12: Chris@12: cout << describe(plugins, pluginBundleBaseURI, describerURI, libname) << endl; Chris@12: Chris@12: return 0; Chris@12: } Chris@12: Chris@12: