diff options
Diffstat (limited to 'src/lib')
20 files changed, 120 insertions, 127 deletions
diff --git a/src/lib/d2srv/d2_simple_parser.cc b/src/lib/d2srv/d2_simple_parser.cc index c57d29edfb..c24dfdcc9a 100644 --- a/src/lib/d2srv/d2_simple_parser.cc +++ b/src/lib/d2srv/d2_simple_parser.cc @@ -286,7 +286,7 @@ void D2SimpleParser::parse(const D2CfgContextPtr& ctx, if (hooks) { HooksLibrariesParser hooks_parser; hooks_parser.parse(libraries, hooks); - libraries.verifyLibraries(hooks->getPosition()); + libraries.verifyLibraries(hooks->getPosition(), false); } // Attempt to create the new client config. This ought to fly as @@ -302,7 +302,7 @@ void D2SimpleParser::parse(const D2CfgContextPtr& ctx, // change causes problems when trying to roll back. HooksManager::prepareUnloadLibraries(); static_cast<void>(HooksManager::unloadLibraries()); - libraries.loadLibraries(); + libraries.loadLibraries(false); } } diff --git a/src/lib/dhcpsrv/cfg_multi_threading.cc b/src/lib/dhcpsrv/cfg_multi_threading.cc index 59a0835a6a..99689742ca 100644 --- a/src/lib/dhcpsrv/cfg_multi_threading.cc +++ b/src/lib/dhcpsrv/cfg_multi_threading.cc @@ -30,7 +30,7 @@ CfgMultiThreading::apply(ConstElementPtr value) { void CfgMultiThreading::extract(ConstElementPtr value, bool& enabled, uint32_t& thread_count, uint32_t& queue_size) { - enabled = false; + enabled = true; thread_count = 0; queue_size = 0; if (value) { diff --git a/src/lib/dhcpsrv/parsers/dhcp_queue_control_parser.cc b/src/lib/dhcpsrv/parsers/dhcp_queue_control_parser.cc index a07e1e2fe2..2169ea2290 100644 --- a/src/lib/dhcpsrv/parsers/dhcp_queue_control_parser.cc +++ b/src/lib/dhcpsrv/parsers/dhcp_queue_control_parser.cc @@ -21,7 +21,8 @@ namespace isc { namespace dhcp { ElementPtr -DHCPQueueControlParser::parse(const ConstElementPtr& control_elem) { +DHCPQueueControlParser::parse(const ConstElementPtr& control_elem, + bool multi_threading_enabled) { // All we really do here is verify that it is a map that // contains at least queue-type. All other content depends // on the packet queue implementation of that type. @@ -47,7 +48,7 @@ DHCPQueueControlParser::parse(const ConstElementPtr& control_elem) { ElementPtr result = data::copy(control_elem); // Currently not compatible with multi-threading. - if (MultiThreadingMgr::instance().getMode()) { + if (multi_threading_enabled) { // Silently disable it. result->set("enable-queue", Element::create(false)); } diff --git a/src/lib/dhcpsrv/parsers/dhcp_queue_control_parser.h b/src/lib/dhcpsrv/parsers/dhcp_queue_control_parser.h index 663874903f..437dd0a049 100644 --- a/src/lib/dhcpsrv/parsers/dhcp_queue_control_parser.h +++ b/src/lib/dhcpsrv/parsers/dhcp_queue_control_parser.h @@ -34,19 +34,20 @@ namespace dhcp { class DHCPQueueControlParser : public isc::data::SimpleParser { public: - /// @brief Constructor - /// + /// @brief Constructor. DHCPQueueControlParser(){}; /// @brief Parses content of the "dhcp-queue-control". /// /// @param control_elem MapElement containing the queue control values to - /// parse + /// parse. + /// @param multi_threading_enabled The flag which indicates if MT is enabled. /// /// @return A copy of the of the input MapElement. /// /// @throw DhcpConfigError if any of the values are invalid. - data::ElementPtr parse(const isc::data::ConstElementPtr& control_elem); + data::ElementPtr parse(const isc::data::ConstElementPtr& control_elem, + bool multi_threading_enabled); private: }; diff --git a/src/lib/dhcpsrv/parsers/multi_threading_config_parser.cc b/src/lib/dhcpsrv/parsers/multi_threading_config_parser.cc index 245caf4e99..8793ba67b9 100644 --- a/src/lib/dhcpsrv/parsers/multi_threading_config_parser.cc +++ b/src/lib/dhcpsrv/parsers/multi_threading_config_parser.cc @@ -28,7 +28,7 @@ MultiThreadingConfigParser::parse(SrvConfig& srv_cfg, } // enable-multi-threading is mandatory - auto enabled = getBoolean(value, "enable-multi-threading"); + getBoolean(value, "enable-multi-threading"); // thread-pool-size is not mandatory if (value->get("thread-pool-size")) { @@ -65,16 +65,6 @@ MultiThreadingConfigParser::parse(SrvConfig& srv_cfg, } srv_cfg.setDHCPMultiThreading(value); - - // Set the mode so that it can be inspected by other configuration parsers - // such as the ones in hook libraries. This creates a dangerous discordance - // between the MT mode and the real configuration. For instance, it may - // result in packets not being processed because the listener thread sees - // the MT mode enabled, but in fact there are 0 threads to handle the - // packets. In production code, it is expected that a call to - // ControlledDhcpvXSrv::processConfig() applies the configuration, which - // should properly create the threads and close this divide. - MultiThreadingMgr::instance().setMode(enabled); } } // namespace dhcp diff --git a/src/lib/dhcpsrv/tests/cfg_multi_threading_unittest.cc b/src/lib/dhcpsrv/tests/cfg_multi_threading_unittest.cc index 3d9de82f65..4cc6676e7b 100644 --- a/src/lib/dhcpsrv/tests/cfg_multi_threading_unittest.cc +++ b/src/lib/dhcpsrv/tests/cfg_multi_threading_unittest.cc @@ -77,7 +77,7 @@ TEST_F(CfgMultiThreadingTest, extract) { //check empty config ASSERT_NO_THROW(CfgMultiThreading::extract(param, enabled, thread_count, queue_size)); - EXPECT_EQ(enabled, false); + EXPECT_EQ(enabled, true); EXPECT_EQ(thread_count, 0); EXPECT_EQ(queue_size, 0); @@ -87,7 +87,7 @@ TEST_F(CfgMultiThreadingTest, extract) { // check empty data ASSERT_NO_THROW(CfgMultiThreading::extract(ConstElementPtr(), enabled, thread_count, queue_size)); - EXPECT_EQ(enabled, false); + EXPECT_EQ(enabled, true); EXPECT_EQ(thread_count, 0); EXPECT_EQ(queue_size, 0); } diff --git a/src/lib/dhcpsrv/tests/dhcp_parsers_unittest.cc b/src/lib/dhcpsrv/tests/dhcp_parsers_unittest.cc index 3d6703d850..50b05b3c5d 100644 --- a/src/lib/dhcpsrv/tests/dhcp_parsers_unittest.cc +++ b/src/lib/dhcpsrv/tests/dhcp_parsers_unittest.cc @@ -91,7 +91,7 @@ TEST_F(DhcpParserTest, MacSources) { EXPECT_NO_THROW(parser.parse(sources, values)); // Finally, check the sources that were configured - CfgMACSources configured_sources = cfg->getMACSources().get(); + CfgMACSources configured_sources = cfg->getMACSources().get(); ASSERT_EQ(2, configured_sources.size()); EXPECT_EQ(HWAddr::HWADDR_SOURCE_DUID, configured_sources[0]); EXPECT_EQ(HWAddr::HWADDR_SOURCE_IPV6_LINK_LOCAL, configured_sources[1]); @@ -230,11 +230,11 @@ public: if (config_pair.first == "hooks-libraries") { HooksLibrariesParser hook_parser; - HooksConfig& libraries = + HooksConfig& libraries = CfgMgr::instance().getStagingCfg()->getHooksConfig(); hook_parser.parse(libraries, config_pair.second); - libraries.verifyLibraries(config_pair.second->getPosition()); - libraries.loadLibraries(); + libraries.verifyLibraries(config_pair.second->getPosition(), false); + libraries.loadLibraries(false); continue; } } @@ -512,8 +512,8 @@ const SimpleDefaults ParseConfigTest::OPTION4_DEF_DEFAULTS = { const SimpleDefaults ParseConfigTest::OPTION6_DEFAULTS = { { "space", Element::string, "dhcp6"}, { "csv-format", Element::boolean, "true"}, - { "always-send", Element::boolean,"false"}, - { "never-send", Element::boolean,"false"} + { "always-send", Element::boolean, "false"}, + { "never-send", Element::boolean, "false"} }; /// This table defines default values for options in DHCPv4 @@ -1697,7 +1697,7 @@ TEST_F(ParseConfigTest, hexOptionData) { "0C 00 03 01 C0 00 03 02", // spaces "0C:00:03:01:C0:00:03:02", // colons "0x0C000301C0000302", // 0x - "C 0 3 1 C0 0 3 02", // one or two digit octets + "C 0 3 1 C0 0 3 02", // one or two digit octets "0x0c000301C0000302" // upper or lower case digits }; @@ -2891,7 +2891,7 @@ TEST_F(ParseConfigTest, defaultSubnet6) { EXPECT_EQ(D2ClientConfig::RCM_NEVER, subnet->getDdnsReplaceClientNameMode().get()); EXPECT_TRUE(subnet->getDdnsGeneratedPrefix().unspecified()); - EXPECT_EQ("", subnet->getDdnsGeneratedPrefix().get()); + EXPECT_TRUE(subnet->getDdnsGeneratedPrefix().empty()); EXPECT_TRUE(subnet->getDdnsQualifyingSuffix().unspecified()); EXPECT_TRUE(subnet->getDdnsQualifyingSuffix().empty()); diff --git a/src/lib/dhcpsrv/tests/dhcp_queue_control_parser_unittest.cc b/src/lib/dhcpsrv/tests/dhcp_queue_control_parser_unittest.cc index e10dccd137..607e907391 100644 --- a/src/lib/dhcpsrv/tests/dhcp_queue_control_parser_unittest.cc +++ b/src/lib/dhcpsrv/tests/dhcp_queue_control_parser_unittest.cc @@ -106,7 +106,7 @@ TEST_F(DHCPQueueControlParserTest, validContent) { // Parsing config into a queue control should succeed. DHCPQueueControlParser parser; try { - queue_control = parser.parse(config_elems); + queue_control = parser.parse(config_elems, false); } catch (const std::exception& ex) { ADD_FAILURE() << "parser threw an exception: " << ex.what(); } @@ -170,7 +170,7 @@ TEST_F(DHCPQueueControlParserTest, invalidContent) { // Parsing config into a queue control should succeed. DHCPQueueControlParser parser; - EXPECT_THROW(parser.parse(config_elems), DhcpConfigError); + EXPECT_THROW(parser.parse(config_elems, false), DhcpConfigError); } } } @@ -193,18 +193,16 @@ TEST_F(DHCPQueueControlParserTest, multiThreading) { // Parse config. DHCPQueueControlParser parser; ConstElementPtr queue_control; - ASSERT_FALSE(MultiThreadingMgr::instance().getMode()); - ASSERT_NO_THROW(queue_control = parser.parse(config_elems)) + ASSERT_NO_THROW(queue_control = parser.parse(config_elems, false)) << "parse fails, test is broken"; + // Verify that queue is enabled. ASSERT_TRUE(queue_control); ASSERT_TRUE(queue_control->get("enable-queue")); EXPECT_EQ("true", queue_control->get("enable-queue")->str()); // Retry with multi-threading. - MultiThreadingTest mt(true); - ASSERT_TRUE(MultiThreadingMgr::instance().getMode()); - ASSERT_NO_THROW(queue_control = parser.parse(config_elems)); + ASSERT_NO_THROW(queue_control = parser.parse(config_elems, true)); ASSERT_TRUE(queue_control); ASSERT_TRUE(queue_control->get("enable-queue")); EXPECT_EQ("false", queue_control->get("enable-queue")->str()); diff --git a/src/lib/dhcpsrv/tests/multi_threading_config_parser_unittest.cc b/src/lib/dhcpsrv/tests/multi_threading_config_parser_unittest.cc index 2272ccbe12..b9196a8b88 100644 --- a/src/lib/dhcpsrv/tests/multi_threading_config_parser_unittest.cc +++ b/src/lib/dhcpsrv/tests/multi_threading_config_parser_unittest.cc @@ -111,8 +111,6 @@ TEST_F(MultiThreadingConfigParserTest, validContent) { CfgMultiThreading::extract(multi_threading_config, enabled, thread_count, queue_size); - EXPECT_EQ(MultiThreadingMgr::instance().getMode(), enabled); - EXPECT_TRUE(multi_threading_config->equals(*config_elems)); } } diff --git a/src/lib/hooks/hooks_config.cc b/src/lib/hooks/hooks_config.cc index c6bf9a2268..6fdd4136f7 100644 --- a/src/lib/hooks/hooks_config.cc +++ b/src/lib/hooks/hooks_config.cc @@ -17,7 +17,8 @@ namespace isc { namespace hooks { void -HooksConfig::verifyLibraries(const Element::Position& position) const { +HooksConfig::verifyLibraries(const Element::Position& position, + bool multi_threading_enabled) const { // The code used to follow this logic: // // Check if the list of libraries has changed. If not, nothing is done @@ -36,7 +37,8 @@ HooksConfig::verifyLibraries(const Element::Position& position) const { // Library list has changed, validate each of the libraries specified. vector<string> lib_names = isc::hooks::extractNames(libraries_); - vector<string> error_libs = HooksManager::validateLibraries(lib_names); + vector<string> error_libs = HooksManager::validateLibraries(lib_names, + multi_threading_enabled); if (!error_libs.empty()) { // Construct the list of libraries in error for the message. @@ -52,12 +54,12 @@ HooksConfig::verifyLibraries(const Element::Position& position) const { } void -HooksConfig::loadLibraries() const { +HooksConfig::loadLibraries(bool multi_threading_enabled) const { /// Commits the list of libraries to the configuration manager storage if /// the list of libraries has changed. /// @todo: Delete any stored CalloutHandles before reloading the /// libraries - if (!HooksManager::loadLibraries(libraries_)) { + if (!HooksManager::loadLibraries(libraries_, multi_threading_enabled)) { isc_throw(InvalidHooksLibraries, "One or more hook libraries failed to load"); } diff --git a/src/lib/hooks/hooks_config.h b/src/lib/hooks/hooks_config.h index 837727e40e..d0cdb67f58 100644 --- a/src/lib/hooks/hooks_config.h +++ b/src/lib/hooks/hooks_config.h @@ -72,8 +72,12 @@ public: /// It tries to validate all the libraries stored in libraries_. /// /// @param position position of the hooks-library map for error reporting + /// @param multi_threading_enabled The flag which indicates if MT is enabled + /// (used to check hook libraries compatibility with MT). + /// /// @throw InvalidHooksLibraries if any issue is discovered. - void verifyLibraries(const isc::data::Element::Position& position) const; + void verifyLibraries(const isc::data::Element::Position& position, + bool multi_threading_enabled) const; /// @brief Commits hooks libraries configuration. /// @@ -83,8 +87,11 @@ public: /// different to those already loaded, this method loads the new set of /// libraries (and unloads the existing set). /// + /// @param multi_threading_enabled The flag which indicates if MT is enabled + /// (used to check hook libraries compatibility with MT). + /// /// @throw InvalidHooksLibraries if the call to HooksManager fails. - void loadLibraries() const; + void loadLibraries(bool multi_threading_enabled) const; /// @brief Unparse a configuration object /// @@ -102,7 +109,7 @@ private: isc::hooks::HookLibsCollection libraries_; }; -}; -}; +} // namespace hooks +} // namespace isc #endif // HOOKS_CONFIG_H diff --git a/src/lib/hooks/hooks_manager.cc b/src/lib/hooks/hooks_manager.cc index 82b741ed25..a5faf76823 100644 --- a/src/lib/hooks/hooks_manager.cc +++ b/src/lib/hooks/hooks_manager.cc @@ -31,7 +31,7 @@ HooksManager::HooksManager() : test_mode_(false) { // libraries, and get the CalloutManager. HookLibsCollection libraries; lm_collection_.reset(new LibraryManagerCollection(libraries)); - lm_collection_->loadLibraries(); + lm_collection_->loadLibraries(false); callout_manager_ = lm_collection_->getCalloutManager(); } @@ -94,7 +94,8 @@ HooksManager::callCommandHandlers(const std::string& command_name, // empty list. bool -HooksManager::loadLibrariesInternal(const HookLibsCollection& libraries) { +HooksManager::loadLibrariesInternal(const HookLibsCollection& libraries, + bool multi_threading_enabled) { if (test_mode_) { return (true); } @@ -114,7 +115,7 @@ HooksManager::loadLibrariesInternal(const HookLibsCollection& libraries) { } // Load the libraries. - bool status = lm_collection_->loadLibraries(); + bool status = lm_collection_->loadLibraries(multi_threading_enabled); if (status) { // ... and obtain the callout manager for them if successful. @@ -129,8 +130,10 @@ HooksManager::loadLibrariesInternal(const HookLibsCollection& libraries) { } bool -HooksManager::loadLibraries(const HookLibsCollection& libraries) { - return (getHooksManager().loadLibrariesInternal(libraries)); +HooksManager::loadLibraries(const HookLibsCollection& libraries, + bool multi_threading_enabled) { + return (getHooksManager().loadLibrariesInternal(libraries, + multi_threading_enabled)); } // Unload the libraries. This just deletes all internal objects (which will @@ -162,7 +165,7 @@ HooksManager::unloadLibrariesInternal() { } // Load the empty set of libraries. - lm_collection_->loadLibraries(); + lm_collection_->loadLibraries(false); // Get the CalloutManager. callout_manager_ = lm_collection_->getCalloutManager(); @@ -255,8 +258,10 @@ HooksManager::postCalloutsLibraryHandle() { // Validate libraries std::vector<std::string> -HooksManager::validateLibraries(const std::vector<std::string>& libraries) { - return (LibraryManagerCollection::validateLibraries(libraries)); +HooksManager::validateLibraries(const std::vector<std::string>& libraries, + bool multi_threading_enabled) { + return (LibraryManagerCollection::validateLibraries(libraries, + multi_threading_enabled)); } // Test mode diff --git a/src/lib/hooks/hooks_manager.h b/src/lib/hooks/hooks_manager.h index 647e14a531..2ca717b0fc 100644 --- a/src/lib/hooks/hooks_manager.h +++ b/src/lib/hooks/hooks_manager.h @@ -66,12 +66,15 @@ public: /// @param libraries List of libraries to be loaded. The order is /// important, as it determines the order that callouts on the same /// hook will be called. + /// @param multi_threading_enabled The flag which indicates if MT is enabled + /// (used to check hook libraries compatibility with MT). /// /// @return true if all libraries loaded without a problem, false if one or /// more libraries failed to load. In the latter case, message will /// be logged that give the reason. /// @throw LibrariesStillOpened when some libraries are already loaded. - static bool loadLibraries(const HookLibsCollection& libraries); + static bool loadLibraries(const HookLibsCollection& libraries, + bool multi_threading_enabled = false); /// @brief Unload libraries /// @@ -247,11 +250,13 @@ public: /// change is committed. /// /// @param libraries List of libraries to be validated. + /// @param multi_threading_enabled The flag which indicates if MT is enabled + /// (used to check hook libraries compatibility with MT). /// /// @return An empty vector if all libraries validated. Otherwise it /// holds the names of the libraries that failed validation. - static std::vector<std::string> validateLibraries( - const std::vector<std::string>& libraries); + static std::vector<std::string> validateLibraries(const std::vector<std::string>& libraries, + bool multi_threading_enabled = false); /// Index numbers for pre-defined hooks. static const int CONTEXT_CREATE = ServerHooks::CONTEXT_CREATE; @@ -379,27 +384,19 @@ private: /// but actually do the work on the singleton instance of the HooksManager. /// See the descriptions of the static methods for more details. - /// @brief Validate library list - /// - /// @param List of libraries to be validated. - /// - /// @return An empty string if all libraries validated. Otherwise it is - /// the name of the first library that failed validation. The - /// configuration code can return this to bindctl as an indication - /// of the problem. - std::string validateLibrariesInternal( - const std::vector<std::string>& libraries) const; - /// @brief Load and reload libraries /// /// @param libraries List of libraries to be loaded. The order is /// important, as it determines the order that callouts on the same /// hook will be called. + /// @param multi_threading_enabled The flag which indicates if MT is enabled + /// (used to check hook libraries compatibility with MT). /// /// @return true if all libraries loaded without a problem, false if one or /// more libraries failed to load. In the latter case, message will /// be logged that give the reason. - bool loadLibrariesInternal(const HookLibsCollection& libraries); + bool loadLibrariesInternal(const HookLibsCollection& libraries, + bool multi_threading_enabled); /// @brief Unload libraries /// diff --git a/src/lib/hooks/library_manager.cc b/src/lib/hooks/library_manager.cc index 447c9ebb88..ac6b109127 100644 --- a/src/lib/hooks/library_manager.cc +++ b/src/lib/hooks/library_manager.cc @@ -139,10 +139,10 @@ LibraryManager::checkVersion() const { // Check the multi-threading compatibility of the library bool -LibraryManager::checkMultiThreadingCompatible() const { +LibraryManager::checkMultiThreadingCompatible(bool multi_threading_enabled) const { // Compatible with single-threaded. - if (!util::MultiThreadingMgr::instance().getMode()) { + if (!multi_threading_enabled) { return (true); } @@ -314,7 +314,7 @@ LibraryManager::prepareUnloadLibrary() { // The main library loading function. bool -LibraryManager::loadLibrary() { +LibraryManager::loadLibrary(bool multi_threading_enabled) { LOG_DEBUG(hooks_logger, HOOKS_DBG_TRACE, HOOKS_LIBRARY_LOADING) .arg(library_name_); @@ -340,7 +340,7 @@ LibraryManager::loadLibrary() { // Library opened OK, see if a version function is present and if so, // check what value it returns. Check multi-threading compatibility. - if (checkVersion() && checkMultiThreadingCompatible()) { + if (checkVersion() && checkMultiThreadingCompatible(multi_threading_enabled)) { // Version OK, so now register the standard callouts and call the // library's load() function if present. registerStandardCallouts(); @@ -407,14 +407,14 @@ LibraryManager::unloadLibrary() { // method. bool -LibraryManager::validateLibrary(const std::string& name) { +LibraryManager::validateLibrary(const std::string& name, bool multi_threading_enabled) { // Instantiate a library manager for the validation. We use the private // constructor as we don't supply a CalloutManager. LibraryManager manager(name); // Try to open it and, if we succeed, check the version. bool validated = manager.openLibrary() && manager.checkVersion() && - manager.checkMultiThreadingCompatible(); + manager.checkMultiThreadingCompatible(multi_threading_enabled); // Regardless of whether the version checked out, close the library. (This // is a no-op if the library failed to open.) diff --git a/src/lib/hooks/library_manager.h b/src/lib/hooks/library_manager.h index f3a376fa62..cbf0880741 100644 --- a/src/lib/hooks/library_manager.h +++ b/src/lib/hooks/library_manager.h @@ -103,10 +103,13 @@ public: /// returns the right number, and the multi-threading compatibility. /// /// @param name Name of the library to validate + /// @param multi_threading_enabled The flag which indicates if MT is enabled + /// (used to check hook libraries compatibility with MT). /// /// @return true if the library validated, false if not. If the library /// fails to validate, the reason for the failure is logged. - static bool validateLibrary(const std::string& name); + static bool validateLibrary(const std::string& name, + bool multi_threading_enabled = false); /// @brief Loads a library /// @@ -119,9 +122,12 @@ public: /// update the global logging dictionary with the log messages /// registered by the loaded library. /// + /// @param multi_threading_enabled The flag which indicates if MT is enabled + /// (used to check hook libraries compatibility with MT). + /// /// @return true if the library loaded successfully, false otherwise. /// In the latter case, the library will be unloaded if possible. - bool loadLibrary(); + bool loadLibrary(bool multi_threading_enabled = false); /// @brief Prepares library unloading /// @@ -192,13 +198,16 @@ protected: /// @brief Check multi-threading compatibility /// - /// If the multi-threading mode is false returns true, else with + /// If the multi-threading mode is disabled returns true, else with /// the library open, accesses the "multi_threading_compatible()" /// function and returns false if not exists or has value 0, returns /// true otherwise. /// + /// @param multi_threading_enabled The flag which indicates if MT is enabled + /// (used to check hook libraries compatibility with MT). + /// /// @return bool true if the check succeeded - bool checkMultiThreadingCompatible() const; + bool checkMultiThreadingCompatible(bool multi_threading_enabled) const; /// @brief Register standard callouts /// diff --git a/src/lib/hooks/library_manager_collection.cc b/src/lib/hooks/library_manager_collection.cc index 6c32b76ce5..50262c1449 100644 --- a/src/lib/hooks/library_manager_collection.cc +++ b/src/lib/hooks/library_manager_collection.cc @@ -50,7 +50,7 @@ LibraryManagerCollection::LibraryManagerCollection(const HookLibsCollection& lib // Load a set of libraries bool -LibraryManagerCollection::loadLibraries() { +LibraryManagerCollection::loadLibraries(bool multi_threading_enabled) { // There must be no libraries still in memory. if (!lib_managers_.empty()) { @@ -90,7 +90,7 @@ LibraryManagerCollection::loadLibraries() { // libraries. On failure, unload all currently loaded libraries, // leaving the object in the state it was in before loadLibraries was // called. - if (manager->loadLibrary()) { + if (manager->loadLibrary(multi_threading_enabled)) { lib_managers_.push_back(manager); } else { static_cast<void>(unloadLibraries()); @@ -136,12 +136,12 @@ LibraryManagerCollection::getLoadedLibraryCount() const { // Validate the libraries. std::vector<std::string> -LibraryManagerCollection::validateLibraries( - const std::vector<std::string>& libraries) { +LibraryManagerCollection::validateLibraries(const std::vector<std::string>& libraries, + bool multi_threading_enabled) { std::vector<std::string> failures; for (size_t i = 0; i < libraries.size(); ++i) { - if (!LibraryManager::validateLibrary(libraries[i])) { + if (!LibraryManager::validateLibrary(libraries[i], multi_threading_enabled)) { failures.push_back(libraries[i]); } } diff --git a/src/lib/hooks/library_manager_collection.h b/src/lib/hooks/library_manager_collection.h index 53a9669a49..1aa28fe07e 100644 --- a/src/lib/hooks/library_manager_collection.h +++ b/src/lib/hooks/library_manager_collection.h @@ -90,9 +90,12 @@ public: /// to load, the loading is abandoned and all libraries loaded so far /// are unloaded. /// + /// @param multi_threading_enabled The flag which indicates if MT is enabled + /// (used to check hook libraries compatibility with MT). + /// /// @return true if all libraries loaded, false if one or more failed t //// load. - bool loadLibraries(); + bool loadLibraries(bool multi_threading_enabled = false); /// @brief Get callout manager /// @@ -137,12 +140,14 @@ public: /// exist, can be opened, that a "version" function is present in them, and /// that it returns the right number. All errors are logged. /// - /// @param libraries List of libraries to validate + /// @param libraries List of libraries to validate. + /// @param multi_threading_enabled The flag which indicates if MT is enabled + /// (used to check hook libraries compatibility with MT). /// /// @return Vector of libraries that failed to validate, or an empty vector /// if all validated. - static std::vector<std::string> - validateLibraries(const std::vector<std::string>& libraries); + static std::vector<std::string> validateLibraries(const std::vector<std::string>& libraries, + bool multi_threading_enabled = false); /// @brief Prepare libaries unloading /// diff --git a/src/lib/hooks/tests/library_manager_unittest.cc b/src/lib/hooks/tests/library_manager_unittest.cc index c0174a41bf..7629c7bf26 100644 --- a/src/lib/hooks/tests/library_manager_unittest.cc +++ b/src/lib/hooks/tests/library_manager_unittest.cc @@ -258,11 +258,10 @@ TEST_F(LibraryManagerTest, NoMultiThreadingCompatible) { EXPECT_TRUE(lib_manager.openLibrary()); // Not multi-threading compatible: does not matter without MT. - EXPECT_TRUE(lib_manager.checkMultiThreadingCompatible()); + EXPECT_TRUE(lib_manager.checkMultiThreadingCompatible(false)); // Not multi-threading compatible: does matter with MT. - MultiThreadingMgr::instance().setMode(true); - EXPECT_FALSE(lib_manager.checkMultiThreadingCompatible()); + EXPECT_FALSE(lib_manager.checkMultiThreadingCompatible(true)); // Tidy up. EXPECT_TRUE(lib_manager.closeLibrary()); @@ -279,11 +278,10 @@ TEST_F(LibraryManagerTest, multiThreadingNotCompatible) { EXPECT_TRUE(lib_manager.openLibrary()); // Not multi-threading compatible: does not matter without MT. - EXPECT_TRUE(lib_manager.checkMultiThreadingCompatible()); + EXPECT_TRUE(lib_manager.checkMultiThreadingCompatible(false)); // Not multi-threading compatible: does matter with MT. - MultiThreadingMgr::instance().setMode(true); - EXPECT_FALSE(lib_manager.checkMultiThreadingCompatible()); + EXPECT_FALSE(lib_manager.checkMultiThreadingCompatible(true)); // Tidy up. EXPECT_TRUE(lib_manager.closeLibrary()); @@ -300,11 +298,10 @@ TEST_F(LibraryManagerTest, multiThreadingCompatible) { EXPECT_TRUE(lib_manager.openLibrary()); // Multi-threading compatible: does not matter without MT. - EXPECT_TRUE(lib_manager.checkMultiThreadingCompatible()); + EXPECT_TRUE(lib_manager.checkMultiThreadingCompatible(false)); // Multi-threading compatible: does matter with MT. - MultiThreadingMgr::instance().setMode(true); - EXPECT_TRUE(lib_manager.checkMultiThreadingCompatible()); + EXPECT_TRUE(lib_manager.checkMultiThreadingCompatible(true)); // Tidy up. EXPECT_TRUE(lib_manager.closeLibrary()); @@ -321,11 +318,10 @@ TEST_F(LibraryManagerTest, multiThreadingCompatibleException) { EXPECT_TRUE(lib_manager.openLibrary()); // Throw exception: does not matter without MT. - EXPECT_TRUE(lib_manager.checkMultiThreadingCompatible()); + EXPECT_TRUE(lib_manager.checkMultiThreadingCompatible(false)); // Throw exception: does matter with MT. - MultiThreadingMgr::instance().setMode(true); - EXPECT_FALSE(lib_manager.checkMultiThreadingCompatible()); + EXPECT_FALSE(lib_manager.checkMultiThreadingCompatible(true)); // Tidy up. EXPECT_TRUE(lib_manager.closeLibrary()); @@ -695,18 +691,16 @@ TEST_F(LibraryManagerTest, validateLibraries) { EXPECT_TRUE(LibraryManager::validateLibrary(UNLOAD_CALLOUT_LIBRARY)); EXPECT_TRUE(LibraryManager::validateLibrary(CALLOUT_PARAMS_LIBRARY)); - MultiThreadingMgr::instance().setMode(true); - - EXPECT_FALSE(LibraryManager::validateLibrary(BASIC_CALLOUT_LIBRARY)); - EXPECT_TRUE(LibraryManager::validateLibrary(FULL_CALLOUT_LIBRARY)); - EXPECT_FALSE(LibraryManager::validateLibrary(FRAMEWORK_EXCEPTION_LIBRARY)); - EXPECT_FALSE(LibraryManager::validateLibrary(INCORRECT_VERSION_LIBRARY)); - EXPECT_FALSE(LibraryManager::validateLibrary(LOAD_CALLOUT_LIBRARY)); - EXPECT_FALSE(LibraryManager::validateLibrary(LOAD_ERROR_CALLOUT_LIBRARY)); - EXPECT_FALSE(LibraryManager::validateLibrary(NOT_PRESENT_LIBRARY)); - EXPECT_FALSE(LibraryManager::validateLibrary(NO_VERSION_LIBRARY)); - EXPECT_FALSE(LibraryManager::validateLibrary(UNLOAD_CALLOUT_LIBRARY)); - EXPECT_FALSE(LibraryManager::validateLibrary(CALLOUT_PARAMS_LIBRARY)); + EXPECT_FALSE(LibraryManager::validateLibrary(BASIC_CALLOUT_LIBRARY, true)); + EXPECT_TRUE(LibraryManager::validateLibrary(FULL_CALLOUT_LIBRARY, true)); + EXPECT_FALSE(LibraryManager::validateLibrary(FRAMEWORK_EXCEPTION_LIBRARY, true)); + EXPECT_FALSE(LibraryManager::validateLibrary(INCORRECT_VERSION_LIBRARY, true)); + EXPECT_FALSE(LibraryManager::validateLibrary(LOAD_CALLOUT_LIBRARY, true)); + EXPECT_FALSE(LibraryManager::validateLibrary(LOAD_ERROR_CALLOUT_LIBRARY, true)); + EXPECT_FALSE(LibraryManager::validateLibrary(NOT_PRESENT_LIBRARY, true)); + EXPECT_FALSE(LibraryManager::validateLibrary(NO_VERSION_LIBRARY, true)); + EXPECT_FALSE(LibraryManager::validateLibrary(UNLOAD_CALLOUT_LIBRARY, true)); + EXPECT_FALSE(LibraryManager::validateLibrary(CALLOUT_PARAMS_LIBRARY, true)); } // Check that log messages are properly registered and unregistered. diff --git a/src/lib/http/client.cc b/src/lib/http/client.cc index 6f41b5ad02..42385ed27e 100644 --- a/src/lib/http/client.cc +++ b/src/lib/http/client.cc @@ -1949,14 +1949,6 @@ private: HttpClient::HttpClient(IOService& io_service, size_t thread_pool_size, bool defer_thread_start /* = false */) { - if (thread_pool_size > 0) { - if (!MultiThreadingMgr::instance().getMode()) { - isc_throw(InvalidOperation, - "HttpClient thread_pool_size must be zero" - "when Kea core multi-threading is disabled"); - } - } - impl_.reset(new HttpClientImpl(io_service, thread_pool_size, defer_thread_start)); } diff --git a/src/lib/http/tests/client_mt_unittests.cc b/src/lib/http/tests/client_mt_unittests.cc index b483273563..e6b98556e6 100644 --- a/src/lib/http/tests/client_mt_unittests.cc +++ b/src/lib/http/tests/client_mt_unittests.cc @@ -844,12 +844,6 @@ TEST_F(MultiThreadingHttpClientTest, basics) { // Make sure destruction doesn't throw. ASSERT_NO_THROW_LOG(client.reset()); - // Non-zero thread-pool-size means multi-threaded mode, should throw. - ASSERT_THROW_MSG(client.reset(new HttpClient(io_service_, 1)), InvalidOperation, - "HttpClient thread_pool_size must be zero" - "when Kea core multi-threading is disabled"); - ASSERT_FALSE(client); - // Enable Kea core multi-threading. MultiThreadingMgr::instance().setMode(true); |