Files
UltraGrid/src/lib_common.cpp
2015-10-27 13:10:54 +01:00

247 lines
9.2 KiB
C++

/**
* @file lib_common.cpp
* @author Martin Pulec <pulec@cesnet.cz>
*/
/*
* Copyright (c) 2012-2015 CESNET, z. s. p. o.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, is permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
*
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* 3. Neither the name of CESNET nor the names of its contributors may be
* used to endorse or promote products derived from this software without
* specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHORS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESSED OR IMPLIED WARRANTIES, INCLUDING,
* BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
* AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
* EVENT SHALL THE AUTHORS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#ifdef HAVE_CONFIG_H
#include "config.h"
#include "config_unix.h"
#include "config_win32.h"
#endif // HAVE_CONFIG_H
#ifdef BUILD_LIBRARIES
#include <dlfcn.h>
#include <glob.h>
#include <libgen.h>
#endif
#include <iostream>
#include <map>
#include "debug.h"
#include "host.h"
#include "lib_common.h"
using namespace std;
struct library_class_info_t {
const char *class_name;
const char *file_prefix;
};
const map<enum library_class, library_class_info_t> library_class_info = {
{ LIBRARY_CLASS_UNDEFINED, { "General module", "" }},
{ LIBRARY_CLASS_CAPTURE_FILTER, { "Video capture filter" , "vcapfilter" }},
{ LIBRARY_CLASS_AUDIO_CAPTURE, { "Audio capture device", "acap" }},
{ LIBRARY_CLASS_AUDIO_PLAYBACK, { "Audio playback device", "aplay" }},
{ LIBRARY_CLASS_VIDEO_CAPTURE, { "Video capture device", "vidcap" }},
{ LIBRARY_CLASS_VIDEO_DISPLAY, { "Video display device", "display" }},
{ LIBRARY_CLASS_AUDIO_COMPRESS, { "Audio compression", "acompress" }},
{ LIBRARY_CLASS_VIDEO_DECOMPRESS, { "Video decompression", "vdecompress" }},
{ LIBRARY_CLASS_VIDEO_COMPRESS, { "Video compression", "vcompress" }},
{ LIBRARY_CLASS_VIDEO_POSTPROCESS, { "Video postprocess", "vo_pp" }},
{ LIBRARY_CLASS_VIDEO_RXTX, { "Video RXTX", "video_rxtx" }},
};
static map<string, string> lib_errors;
void open_all(const char *pattern) {
#ifdef BUILD_LIBRARIES
char path[512];
glob_t glob_buf;
char *tmp = strdup(uv_argv[0]);
/* binary not from $PATH */
if(strchr(uv_argv[0], '/') != NULL) {
char *dir = dirname(tmp);
snprintf(path, sizeof(path), "%s/../lib/ultragrid/%s", dir, pattern);
} else {
snprintf(path, sizeof(path), TOSTRING(LIB_DIR) "/ultragrid/%s", pattern);
}
free(tmp);
glob(path, 0, NULL, &glob_buf);
for(unsigned int i = 0; i < glob_buf.gl_pathc; ++i) {
if (!dlopen(glob_buf.gl_pathv[i], RTLD_NOW|RTLD_GLOBAL)) {
char *error = dlerror();
verbose_msg("Library %s opening warning: %s \n", glob_buf.gl_pathv[i],
error);
char *filename = basename(glob_buf.gl_pathv[i]);
if (filename && error) {
lib_errors.emplace(filename, error);
}
}
}
globfree(&glob_buf);
#else
UNUSED(pattern);
#endif
}
struct lib_info {
const void *data;
int abi_version;
};
// http://stackoverflow.com/questions/1801892/making-mapfind-operation-case-insensitive
/************************************************************************/
/* Comparator for case-insensitive comparison in STL assos. containers */
/************************************************************************/
struct ci_less : std::binary_function<std::string, std::string, bool>
{
// case-independent (ci) compare_less binary function
struct nocase_compare : public std::binary_function<unsigned char,unsigned char,bool>
{
bool operator() (const unsigned char& c1, const unsigned char& c2) const {
return tolower (c1) < tolower (c2);
}
};
bool operator() (const std::string & s1, const std::string & s2) const {
return std::lexicographical_compare
(s1.begin (), s1.end (), // source range
s2.begin (), s2.end (), // dest range
nocase_compare ()); // comparison
}
};
static map<enum library_class, map<string, lib_info, ci_less>> *libraries = nullptr;
/**
* The purpose of this initializor instead of ordinary static initialization is that register_video_capture_filter()
* may be called before static members are initialized (it is __attribute__((constructor)))
*/
struct init_libraries {
init_libraries() {
if (libraries == nullptr) {
libraries = new remove_pointer<decltype(libraries)>::type();
}
}
};
static struct init_libraries loader;
void register_library(const char *name, const void *data, enum library_class cls, int abi_version)
{
struct init_libraries loader;
(*libraries)[cls][name] = {data, abi_version};
}
const void *load_library(const char *name, enum library_class cls, int abi_version)
{
if (libraries->find(cls) != libraries->end()) {
auto it_cls = libraries->find(cls)->second;
auto it_module = it_cls.find(name);
if (it_module != it_cls.end()) {
auto mod_pair = it_cls.find(name)->second;
if (mod_pair.abi_version == abi_version) {
return mod_pair.data;
} else {
LOG(LOG_LEVEL_WARNING) << "Module " << name << " ABI version mismatch (required " <<
abi_version << ", have " << mod_pair.abi_version << ")\n";
}
}
}
// Library was not found or was not loaded due to unsatisfied
// dependencies. If the latter one, display reason why dlopen() failed.
if (library_class_info.find(cls) != library_class_info.end()) {
string filename = "module_";
if (strlen(library_class_info.at(cls).file_prefix) > 0) {
filename += library_class_info.at(cls).file_prefix;
filename += "_";
}
filename += name + string(".so");
if (lib_errors.find(filename) != lib_errors.end()) {
LOG(LOG_LEVEL_WARNING) << filename << ": " << lib_errors.find(filename)->second << "\n";
}
}
return NULL;
}
void list_modules(enum library_class cls, int abi_version) {
const auto & class_set = get_libraries_for_class(cls, abi_version);
for (auto && item : class_set) {
printf("\t%s\n", item.first.c_str());
}
}
void list_all_modules() {
for (auto cls_it = library_class_info.begin(); cls_it != library_class_info.end();
++cls_it) {
cout << cls_it->second.class_name << "\n";
auto it = libraries->find(cls_it->first);
if (it != libraries->end()) {
for (auto && item : it->second) {
cout << "\t" << item.first << "\n";
}
}
cout << "\n";
}
if (!lib_errors.empty()) {
cout << "Errors:\n";
for (auto && item : lib_errors) {
cout << "\t" << item.first << "\n\t\t" << item.second << "\n";
}
cout << "\n";
}
}
map<string, const void *> get_libraries_for_class(enum library_class cls, int abi_version)
{
map<string, const void *> ret;
auto it = libraries->find(cls);
if (it != libraries->end()) {
for (auto && item : it->second) {
if (abi_version == item.second.abi_version) {
ret[item.first] = item.second.data;
} else {
LOG(LOG_LEVEL_WARNING) << "Module " << item.first << " ABI version mismatch (required " <<
abi_version << ", have " << item.second.abi_version << ")\n";
}
}
}
return ret;
}