Mercurial > hg > vamp-plugin-sdk
diff vamp-sdk/PluginAdapter.cpp @ 12:a3d35e11c3fe
* Avoid repeated malloc/free for returned feature lists -- reuse static
feature lists where possible. Need to document the host behaviour that
permits this (i.e. a returned feature list is only valid until the next
call to process, getRemainingFeatures or releaseFeatureSet)
author | cannam |
---|---|
date | Thu, 06 Apr 2006 15:12:25 +0000 |
parents | 44113b1e296b |
children | 85801331454c |
line wrap: on
line diff
--- a/vamp-sdk/PluginAdapter.cpp Thu Apr 06 12:26:44 2006 +0000 +++ b/vamp-sdk/PluginAdapter.cpp Thu Apr 06 15:12:25 2006 +0000 @@ -309,7 +309,7 @@ free((void *)desc); } -VampFeatureList ** +VampFeatureList * PluginAdapterBase::vampProcess(VampPluginHandle handle, float **inputBuffers, int sec, @@ -321,7 +321,7 @@ inputBuffers, sec, nsec); } -VampFeatureList ** +VampFeatureList * PluginAdapterBase::vampGetRemainingFeatures(VampPluginHandle handle) { PluginAdapterBase *adapter = lookupAdapter(handle); @@ -330,28 +330,35 @@ } void -PluginAdapterBase::vampReleaseFeatureSet(VampFeatureList **fs) +PluginAdapterBase::vampReleaseFeatureSet(VampFeatureList *fs) { - if (!fs) return; - - for (unsigned int i = 0; fs[i]; ++i) { - - for (unsigned int j = 0; j < fs[i]->featureCount; ++j) { - VampFeature *feature = &fs[i]->features[j]; - if (feature->values) free((void *)feature->values); - if (feature->label) free((void *)feature->label); - } - - if (fs[i]->features) free((void *)fs[i]->features); - free((void *)fs[i]); - } - - free((void *)fs); } void PluginAdapterBase::cleanup(Plugin *plugin) { + if (m_fs.find(plugin) != m_fs.end()) { + size_t outputCount = 0; + if (m_pluginOutputs[plugin]) { + outputCount = m_pluginOutputs[plugin]->size(); + } + VampFeatureList *list = m_fs[plugin]; + for (unsigned int i = 0; i < outputCount; ++i) { + for (unsigned int j = 0; j < m_fsizes[plugin][i]; ++j) { + if (list[i].features[j].label) { + free(list[i].features[j].label); + } + if (list[i].features[j].values) { + free(list[i].features[j].values); + } + } + if (list[i].features) free(list[i].features); + } + m_fs.erase(plugin); + m_fsizes.erase(plugin); + m_fvsizes.erase(plugin); + } + if (m_pluginOutputs.find(plugin) != m_pluginOutputs.end()) { delete m_pluginOutputs[plugin]; m_pluginOutputs.erase(plugin); @@ -428,89 +435,167 @@ return desc; } -VampFeatureList ** +VampFeatureList * PluginAdapterBase::process(Plugin *plugin, float **inputBuffers, int sec, int nsec) { +// std::cerr << "PluginAdapterBase::process" << std::endl; RealTime rt(sec, nsec); - return convertFeatures(plugin->process(inputBuffers, rt)); + checkOutputMap(plugin); + return convertFeatures(plugin, plugin->process(inputBuffers, rt)); } -VampFeatureList ** +VampFeatureList * PluginAdapterBase::getRemainingFeatures(Plugin *plugin) { - return convertFeatures(plugin->getRemainingFeatures()); +// std::cerr << "PluginAdapterBase::getRemainingFeatures" << std::endl; + checkOutputMap(plugin); + return convertFeatures(plugin, plugin->getRemainingFeatures()); } -VampFeatureList ** -PluginAdapterBase::convertFeatures(const Plugin::FeatureSet &features) +VampFeatureList * +PluginAdapterBase::convertFeatures(Plugin *plugin, + const Plugin::FeatureSet &features) { - unsigned int n = 0; - if (features.begin() != features.end()) { - Plugin::FeatureSet::const_iterator i = features.end(); - --i; - n = i->first + 1; - } + int lastN = -1; - if (!n) return 0; + int outputCount = 0; + if (m_pluginOutputs[plugin]) outputCount = m_pluginOutputs[plugin]->size(); + + resizeFS(plugin, outputCount); + VampFeatureList *fs = m_fs[plugin]; - VampFeatureList **fs = (VampFeatureList **) - malloc((n + 1) * sizeof(VampFeatureList *)); + for (Plugin::FeatureSet::const_iterator fi = features.begin(); + fi != features.end(); ++fi) { - for (unsigned int i = 0; i < n; ++i) { + int n = fi->first; + +// std::cerr << "PluginAdapterBase::convertFeatures: n = " << n << std::endl; - fs[i] = (VampFeatureList *)malloc(sizeof(VampFeatureList)); - - if (features.find(i) == features.end()) { - - fs[i]->featureCount = 0; - fs[i]->features = 0; + if (n >= int(outputCount)) { + std::cerr << "WARNING: PluginAdapterBase::convertFeatures: Too many outputs from plugin (" << n+1 << ", only should be " << outputCount << ")" << std::endl; continue; } - Plugin::FeatureSet::const_iterator fi = features.find(i); + if (n > lastN + 1) { + for (int i = lastN + 1; i < n; ++i) { + fs[i].featureCount = 0; + } + } const Plugin::FeatureList &fl = fi->second; - fs[i]->featureCount = fl.size(); + size_t sz = fl.size(); + if (sz > m_fsizes[plugin][n]) resizeFL(plugin, n, sz); + fs[n].featureCount = sz; + + for (size_t j = 0; j < sz; ++j) { - if (fs[i]->featureCount == 0) { - fs[i]->features = 0; - continue; - } +// std::cerr << "PluginAdapterBase::convertFeatures: j = " << j << std::endl; - fs[i]->features = (VampFeature *)malloc(fl.size() * sizeof(VampFeature)); - - for (unsigned int j = 0; j < fl.size(); ++j) { - - VampFeature *feature = &fs[i]->features[j]; + VampFeature *feature = &fs[n].features[j]; feature->hasTimestamp = fl[j].hasTimestamp; feature->sec = fl[j].timestamp.sec; feature->nsec = fl[j].timestamp.nsec; feature->valueCount = fl[j].values.size(); - feature->label = strdup(fl[j].label.c_str()); - if (feature->valueCount == 0) { - feature->values = 0; - continue; + if (feature->label) free(feature->label); + + if (fl[j].label.empty()) { + feature->label = 0; + } else { + feature->label = strdup(fl[j].label.c_str()); } - feature->values = (float *)malloc - (feature->valueCount * sizeof(float)); + if (feature->valueCount > m_fvsizes[plugin][n][j]) { + resizeFV(plugin, n, j, feature->valueCount); + } for (unsigned int k = 0; k < feature->valueCount; ++k) { +// std::cerr << "PluginAdapterBase::convertFeatures: k = " << k << std::endl; feature->values[k] = fl[j].values[k]; } } + + lastN = n; } - fs[n] = 0; + if (lastN == -1) return 0; + + if (int(outputCount) > lastN + 1) { + for (int i = lastN + 1; i < int(outputCount); ++i) { + fs[i].featureCount = 0; + } + } return fs; } +void +PluginAdapterBase::resizeFS(Plugin *plugin, int n) +{ +// std::cerr << "PluginAdapterBase::resizeFS(" << plugin << ", " << n << ")" << std::endl; + + int i = m_fsizes[plugin].size(); + if (i >= n) return; + + std::cerr << "resizing from " << i << std::endl; + + m_fs[plugin] = (VampFeatureList *)realloc + (m_fs[plugin], n * sizeof(VampFeatureList)); + + while (i < n) { + m_fs[plugin][i].featureCount = 0; + m_fs[plugin][i].features = 0; + m_fsizes[plugin].push_back(0); + m_fvsizes[plugin].push_back(std::vector<size_t>()); + i++; + } +} + +void +PluginAdapterBase::resizeFL(Plugin *plugin, int n, size_t sz) +{ + std::cerr << "PluginAdapterBase::resizeFL(" << plugin << ", " << n << ", " + << sz << ")" << std::endl; + + size_t i = m_fsizes[plugin][n]; + if (i >= sz) return; + + std::cerr << "resizing from " << i << std::endl; + + m_fs[plugin][n].features = (VampFeature *)realloc + (m_fs[plugin][n].features, sz * sizeof(VampFeature)); + + while (m_fsizes[plugin][n] < sz) { + m_fs[plugin][n].features[m_fsizes[plugin][n]].valueCount = 0; + m_fs[plugin][n].features[m_fsizes[plugin][n]].values = 0; + m_fs[plugin][n].features[m_fsizes[plugin][n]].label = 0; + m_fvsizes[plugin][n].push_back(0); + m_fsizes[plugin][n]++; + } +} + +void +PluginAdapterBase::resizeFV(Plugin *plugin, int n, int j, size_t sz) +{ + + std::cerr << "PluginAdapterBase::resizeFV(" << plugin << ", " << n << ", " + << j << ", " << sz << ")" << std::endl; + + size_t i = m_fvsizes[plugin][n][j]; + if (i >= sz) return; + + std::cerr << "resizing from " << i << std::endl; + + m_fs[plugin][n].features[j].values = (float *)realloc + (m_fs[plugin][n].features[j].values, sz * sizeof(float)); + + m_fvsizes[plugin][n][j] = sz; +} + PluginAdapterBase::AdapterMap PluginAdapterBase::m_adapterMap;