/* * Copyright (c) 2011-2014, Intel Corporation * All rights reserved. * * Redistribution and use in source and binary forms, with or without modification, * are 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 the copyright holder 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 COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND * ANY EXPRESS 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 COPYRIGHT HOLDER 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. */ #include <dlfcn.h> #include <dirent.h> #include <algorithm> #include <ctype.h> #include "SystemClass.h" #include "SubsystemLibrary.h" #include "AutoLog.h" #include "VirtualSubsystem.h" #include "NamedElementBuilderTemplate.h" #include <assert.h> #include "PluginLocation.h" #include "Utility.h" #define base CConfigurableElement using std::list; using std::string; /** * A plugin file name is of the form: * lib<type>-subsystem.so or lib<type>-subsystem._host.so * * The plugin symbol is of the form: * get<TYPE>SubsystemBuilder */ // Plugin file naming const char* gpcPluginSuffix = "-subsystem"; const char* gpcPluginPrefix = "lib"; // Plugin symbol naming const char* gpcPluginSymbolPrefix = "get"; const char* gpcPluginSymbolSuffix = "SubsystemBuilder"; // Used by subsystem plugins typedef void (*GetSubsystemBuilder)(CSubsystemLibrary*); CSystemClass::CSystemClass() : _pSubsystemLibrary(new CSubsystemLibrary) { } CSystemClass::~CSystemClass() { delete _pSubsystemLibrary; // Destroy child subsystems *before* unloading the libraries (otherwise crashes will occur // as unmapped code will be referenced) clean(); // Close all previously opened subsystem libraries list<void*>::const_iterator it; for (it = _subsystemLibraryHandleList.begin(); it != _subsystemLibraryHandleList.end(); ++it) { dlclose(*it); } } bool CSystemClass::childrenAreDynamic() const { return true; } string CSystemClass::getKind() const { return "SystemClass"; } bool CSystemClass::loadSubsystems(string& strError, const CSubsystemPlugins* pSubsystemPlugins, bool bVirtualSubsystemFallback) { CAutoLog autoLog_info(this, "Loading subsystem plugins"); // Start clean _pSubsystemLibrary->clean(); // Add virtual subsystem builder _pSubsystemLibrary->addElementBuilder("Virtual", new TNamedElementBuilderTemplate<CVirtualSubsystem>()); // Set virtual subsytem as builder fallback if required _pSubsystemLibrary->enableDefaultMechanism(bVirtualSubsystemFallback); // Add subsystem defined in shared libraries list<string> lstrError; bool bLoadPluginsSuccess = loadSubsystemsFromSharedLibraries(lstrError, pSubsystemPlugins); if (bLoadPluginsSuccess) { log_info("All subsystem plugins successfully loaded"); } else { // Log plugin as warning if no fallback available log_table(!bVirtualSubsystemFallback, lstrError); } if (!bVirtualSubsystemFallback) { // Any problem reported is an error as there is no fallback. // Fill strError for caller. CUtility::asString(lstrError, strError); } return bLoadPluginsSuccess || bVirtualSubsystemFallback; } bool CSystemClass::loadSubsystemsFromSharedLibraries(list<string>& lstrError, const CSubsystemPlugins* pSubsystemPlugins) { // Plugin list list<string> lstrPluginFiles; uint32_t uiPluginLocation; for (uiPluginLocation = 0; uiPluginLocation < pSubsystemPlugins->getNbChildren(); uiPluginLocation++) { // Get Folder for current Plugin Location const CPluginLocation* pPluginLocation = static_cast<const CPluginLocation*>(pSubsystemPlugins->getChild(uiPluginLocation)); string strFolder(pPluginLocation->getFolder()); if (!strFolder.empty()) { strFolder += "/"; } // Iterator on Plugin List: list<string>::const_iterator it; const list<string>& pluginList = pPluginLocation->getPluginList(); for (it = pluginList.begin(); it != pluginList.end(); ++it) { // Fill Plugin files list lstrPluginFiles.push_back(strFolder + *it); } } // Actually load plugins while (!lstrPluginFiles.empty()) { // Because plugins might depend on one another, loading will be done // as an iteration process that finishes successfully when the remaining // list of plugins to load gets empty or unsuccessfully if the loading // process failed to load at least one of them // Attempt to load the complete list if (!loadPlugins(lstrPluginFiles, lstrError)) { // Unable to load at least one plugin break; } } if (!lstrPluginFiles.empty()) { // Unable to load at least one plugin string strPluginUnloaded; CUtility::asString(lstrPluginFiles, strPluginUnloaded, ", "); lstrError.push_back("Unable to load the following plugins: " + strPluginUnloaded + "."); return false; } return true; } // Plugin symbol computation string CSystemClass::getPluginSymbol(const string& strPluginPath) { // Extract plugin type out of file name string strPluginSuffix = gpcPluginSuffix; string strPluginPrefix = gpcPluginPrefix; // Remove folder and library prefix size_t iPluginTypePos = strPluginPath.rfind('/') + 1 + strPluginPrefix.length(); // Get index of -subsystem.so or -subsystem_host.so suffix size_t iSubsystemPos = strPluginPath.find(strPluginSuffix, iPluginTypePos); // Get type (between iPluginTypePos and iSubsystemPos) string strPluginType = strPluginPath.substr(iPluginTypePos, iSubsystemPos - iPluginTypePos); // Make it upper case std::transform(strPluginType.begin(), strPluginType.end(), strPluginType.begin(), ::toupper); // Get plugin symbol return gpcPluginSymbolPrefix + strPluginType + gpcPluginSymbolSuffix; } // Plugin loading bool CSystemClass::loadPlugins(list<string>& lstrPluginFiles, list<string>& lstrError) { assert(lstrPluginFiles.size()); bool bAtLeastOneSubsystemPluginSuccessfullyLoaded = false; list<string>::iterator it = lstrPluginFiles.begin(); while (it != lstrPluginFiles.end()) { string strPluginFileName = *it; log_info("Attempting to load subsystem plugin path \"%s\"", strPluginFileName.c_str()); // Load attempt void* lib_handle = dlopen(strPluginFileName.c_str(), RTLD_LAZY); if (!lib_handle) { const char *err = dlerror(); // Failed if (err == NULL) { lstrError.push_back("dlerror failed"); } else { lstrError.push_back("Plugin load failed: " + string(err)); } // Next plugin ++it; continue; } // Store libraries handles _subsystemLibraryHandleList.push_back(lib_handle); // Get plugin symbol string strPluginSymbol = getPluginSymbol(strPluginFileName); // Load symbol from library GetSubsystemBuilder pfnGetSubsystemBuilder = (GetSubsystemBuilder)dlsym(lib_handle, strPluginSymbol.c_str()); if (!pfnGetSubsystemBuilder) { lstrError.push_back("Subsystem plugin " + strPluginFileName + " does not contain " + strPluginSymbol + " symbol."); continue; } // Account for this success bAtLeastOneSubsystemPluginSuccessfullyLoaded = true; // Fill library pfnGetSubsystemBuilder(_pSubsystemLibrary); // Remove successfully loaded plugin from list and select next lstrPluginFiles.erase(it++); } return bAtLeastOneSubsystemPluginSuccessfullyLoaded; } const CSubsystemLibrary* CSystemClass::getSubsystemLibrary() const { return _pSubsystemLibrary; } void CSystemClass::checkForSubsystemsToResync(CSyncerSet& syncerSet) { size_t uiNbChildren = getNbChildren(); size_t uiChild; for (uiChild = 0; uiChild < uiNbChildren; uiChild++) { CSubsystem* pSubsystem = static_cast<CSubsystem*>(getChild(uiChild)); // Collect and consume the need for a resync if (pSubsystem->needResync(true)) { log_info("Resynchronizing subsystem: %s", pSubsystem->getName().c_str()); // get all subsystem syncers pSubsystem->fillSyncerSet(syncerSet); } } } void CSystemClass::cleanSubsystemsNeedToResync() { size_t uiNbChildren = getNbChildren(); size_t uiChild; for (uiChild = 0; uiChild < uiNbChildren; uiChild++) { CSubsystem* pSubsystem = static_cast<CSubsystem*>(getChild(uiChild)); // Consume the need for a resync pSubsystem->needResync(true); } }