summaryrefslogtreecommitdiffstats
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/bin/dhcp4/tests/callout_library_1.cc14
-rw-r--r--src/bin/dhcp4/tests/callout_library_3.cc10
-rw-r--r--src/bin/dhcp4/tests/ctrl_dhcp4_srv_unittest.cc51
-rw-r--r--src/bin/dhcp4/tests/hooks_unittest.cc53
-rw-r--r--src/bin/dhcp6/tests/callout_library_1.cc14
-rw-r--r--src/bin/dhcp6/tests/callout_library_3.cc10
-rw-r--r--src/bin/dhcp6/tests/ctrl_dhcp6_srv_unittest.cc56
-rw-r--r--src/bin/dhcp6/tests/hooks_unittest.cc51
8 files changed, 254 insertions, 5 deletions
diff --git a/src/bin/dhcp4/tests/callout_library_1.cc b/src/bin/dhcp4/tests/callout_library_1.cc
index 3525aa9bbc..73e3a17f90 100644
--- a/src/bin/dhcp4/tests/callout_library_1.cc
+++ b/src/bin/dhcp4/tests/callout_library_1.cc
@@ -14,3 +14,17 @@ static const int LIBRARY_NUMBER = 1;
#include <config.h>
#include <dhcp4/tests/callout_library_common.h>
+
+// Functions accessed by the hooks framework use C linkage to avoid the name
+// mangling that accompanies use of the C++ compiler as well as to avoid
+// issues related to namespaces.
+extern "C" {
+
+/// @brief This function is called to retrieve the multi-threading compatibility.
+///
+/// @return 1 which means compatible with multi-threading.
+int multi_threading_compatible() {
+ return (1);
+}
+
+} // end extern "C"
diff --git a/src/bin/dhcp4/tests/callout_library_3.cc b/src/bin/dhcp4/tests/callout_library_3.cc
index 6c69ae7442..8b6668eac0 100644
--- a/src/bin/dhcp4/tests/callout_library_3.cc
+++ b/src/bin/dhcp4/tests/callout_library_3.cc
@@ -17,6 +17,9 @@ static const int LIBRARY_NUMBER = 3;
using namespace isc::hooks;
+// Functions accessed by the hooks framework use C linkage to avoid the name
+// mangling that accompanies use of the C++ compiler as well as to avoid
+// issues related to namespaces.
extern "C" {
/// @brief Callout which appends library number and provided arguments to
@@ -43,4 +46,11 @@ dhcp4_srv_configured(CalloutHandle& handle) {
return (0);
}
+/// @brief This function is called to retrieve the multi-threading compatibility.
+///
+/// @return 0 which means not compatible with multi-threading.
+int multi_threading_compatible() {
+ return (0);
}
+
+} // end extern "C"
diff --git a/src/bin/dhcp4/tests/ctrl_dhcp4_srv_unittest.cc b/src/bin/dhcp4/tests/ctrl_dhcp4_srv_unittest.cc
index 6556dd20a6..753df545ef 100644
--- a/src/bin/dhcp4/tests/ctrl_dhcp4_srv_unittest.cc
+++ b/src/bin/dhcp4/tests/ctrl_dhcp4_srv_unittest.cc
@@ -22,6 +22,7 @@
#include <log/logger_support.h>
#include <stats/stats_mgr.h>
#include <util/boost_time_utils.h>
+#include <util/multi_threading_mgr.h>
#include <testutils/io_utils.h>
#include <testutils/unix_control_client.h>
#include <testutils/sandbox.h>
@@ -52,6 +53,7 @@ using namespace isc::dhcp::test;
using namespace isc::hooks;
using namespace isc::stats;
using namespace isc::test;
+using namespace isc::util;
namespace {
@@ -115,6 +117,7 @@ public:
socket_path_ = sandbox.join("kea4.sock");
}
reset();
+ MultiThreadingMgr::instance().setMode(false);
}
/// @brief Destructor
@@ -127,6 +130,7 @@ public:
CommandMgr::instance().setConnectionTimeout(TIMEOUT_DHCP_SERVER_RECEIVE_COMMAND);
server_.reset();
+ MultiThreadingMgr::instance().setMode(false);
};
/// @brief Returns pointer to the server's IO service.
@@ -457,6 +461,53 @@ TEST_F(CtrlChannelDhcpv4SrvTest, libreload) {
EXPECT_TRUE(checkMarkerFile(LOAD_MARKER_FILE, "1212"));
}
+// Check that the "libreload" command will reload libraries when MT is enabled
+TEST_F(CtrlChannelDhcpv4SrvTest, libreloadFailMultiThreading) {
+ createUnixChannelServer();
+
+ // Ensure no marker files to start with.
+ ASSERT_FALSE(checkMarkerFileExists(LOAD_MARKER_FILE));
+ ASSERT_FALSE(checkMarkerFileExists(UNLOAD_MARKER_FILE));
+
+ // Load two libraries
+ HookLibsCollection libraries;
+ libraries.push_back(make_pair(CALLOUT_LIBRARY_1, ConstElementPtr()));
+ libraries.push_back(make_pair(CALLOUT_LIBRARY_2, ConstElementPtr()));
+ HooksManager::loadLibraries(libraries);
+
+ // Check they are loaded.
+ HookLibsCollection loaded_libraries =
+ HooksManager::getLibraryInfo();
+ ASSERT_TRUE(libraries == loaded_libraries);
+
+ // ... which also included checking that the marker file created by the
+ // load functions exists and holds the correct value (of "12" - the
+ // first library appends "1" to the file, the second appends "2"). Also
+ // check that the unload marker file does not yet exist.
+ EXPECT_TRUE(checkMarkerFile(LOAD_MARKER_FILE, "12"));
+ EXPECT_FALSE(checkMarkerFileExists(UNLOAD_MARKER_FILE));
+
+ // Enable multi-threading before libreload command which should now fail
+ // as the second library is not multi-threading compatible.
+ MultiThreadingMgr::instance().setMode(true);
+
+ // Now execute the "libreload" command. This should cause the libraries
+ // to unload and to reload.
+ std::string response;
+ sendUnixCommand("{ \"command\": \"libreload\" }", response);
+ EXPECT_EQ("{ \"result\": 1, "
+ "\"text\": \"Failed to reload hooks libraries.\" }"
+ , response);
+
+ // Check that the libraries have unloaded and failed to reload. The
+ // libraries are unloaded in the reverse order to which they are loaded.
+ // When they load, they should append information to the loading marker
+ // file. Failing to load the second library will also unload the first
+ // library.
+ EXPECT_TRUE(checkMarkerFile(UNLOAD_MARKER_FILE, "211"));
+ EXPECT_TRUE(checkMarkerFile(LOAD_MARKER_FILE, "121"));
+}
+
// This test checks which commands are registered by the DHCPv4 server.
TEST_F(CtrlChannelDhcpv4SrvTest, commandsRegistration) {
diff --git a/src/bin/dhcp4/tests/hooks_unittest.cc b/src/bin/dhcp4/tests/hooks_unittest.cc
index 179d6d32d5..8f83e824ab 100644
--- a/src/bin/dhcp4/tests/hooks_unittest.cc
+++ b/src/bin/dhcp4/tests/hooks_unittest.cc
@@ -23,6 +23,7 @@
#include <dhcp4/tests/dhcp4_client.h>
#include <dhcp4/tests/marker_file.h>
#include <dhcp4/tests/test_libraries.h>
+#include <util/multi_threading_mgr.h>
#include <vector>
@@ -33,6 +34,7 @@ using namespace isc::hooks;
using namespace isc::config;
using namespace isc::dhcp::test;
using namespace isc::dhcp;
+using namespace isc::util;
// Checks if hooks are registered properly.
TEST_F(Dhcpv4SrvTest, Hooks) {
@@ -860,12 +862,14 @@ public:
LoadUnloadDhcpv4SrvTest() {
reset();
+ MultiThreadingMgr::instance().setMode(false);
}
/// @brief Destructor
~LoadUnloadDhcpv4SrvTest() {
server_.reset();
reset();
+ MultiThreadingMgr::instance().setMode(false);
};
/// @brief Reset hooks data
@@ -2730,8 +2734,8 @@ TEST_F(LoadUnloadDhcpv4SrvTest, unloadLibraries) {
libraries.push_back(make_pair(std::string(CALLOUT_LIBRARY_1),
ConstElementPtr()));
libraries.push_back(make_pair(std::string(CALLOUT_LIBRARY_2),
-
ConstElementPtr()));
+
ASSERT_TRUE(HooksManager::loadLibraries(libraries));
// Verify that they load functions created the LOAD_MARKER_FILE
@@ -2751,6 +2755,53 @@ TEST_F(LoadUnloadDhcpv4SrvTest, unloadLibraries) {
EXPECT_TRUE(checkMarkerFile(LOAD_MARKER_FILE, "12"));
}
+// Verifies that libraries incompatible with multi threading are not loaded by
+// the server
+// The callout libraries write their library index number to a marker
+// file upon load and unload, making it simple to test whether or not
+// the load and unload callouts have been invoked.
+TEST_F(LoadUnloadDhcpv4SrvTest, failLoadIncompatibleLibraries) {
+
+ ASSERT_NO_THROW(server_.reset(new NakedDhcpv4Srv()));
+
+ MultiThreadingMgr::instance().setMode(true);
+
+ // Ensure no marker files to start with.
+ ASSERT_FALSE(checkMarkerFileExists(LOAD_MARKER_FILE));
+ ASSERT_FALSE(checkMarkerFileExists(UNLOAD_MARKER_FILE));
+
+ // Load the test libraries
+ HookLibsCollection libraries;
+ libraries.push_back(make_pair(std::string(CALLOUT_LIBRARY_2),
+ ConstElementPtr()));
+
+ ASSERT_FALSE(HooksManager::loadLibraries(libraries));
+
+ // The library is missing multi_threading_compatible function so loading
+ // should fail
+ EXPECT_FALSE(checkMarkerFileExists(LOAD_MARKER_FILE));
+ EXPECT_FALSE(checkMarkerFileExists(UNLOAD_MARKER_FILE));
+
+ libraries.clear();
+ libraries.push_back(make_pair(std::string(CALLOUT_LIBRARY_3),
+ ConstElementPtr()));
+
+ ASSERT_FALSE(HooksManager::loadLibraries(libraries));
+
+ // The library is not multi threading compatible so loading should fail
+ EXPECT_FALSE(checkMarkerFileExists(LOAD_MARKER_FILE));
+ EXPECT_FALSE(checkMarkerFileExists(UNLOAD_MARKER_FILE));
+
+ // Destroy the server, instance which should unload the libraries.
+ server_.reset();
+
+ // Check that the libraries were unloaded. The libraries are
+ // unloaded in the reverse order to which they are loaded, and
+ // this should be reflected in the unload file.
+ EXPECT_FALSE(checkMarkerFileExists(LOAD_MARKER_FILE));
+ EXPECT_FALSE(checkMarkerFileExists(UNLOAD_MARKER_FILE));
+}
+
// Checks if callouts installed on the dhcp4_srv_configured ared indeed called
// and all the necessary parameters are passed.
TEST_F(LoadUnloadDhcpv4SrvTest, Dhcpv4SrvConfigured) {
diff --git a/src/bin/dhcp6/tests/callout_library_1.cc b/src/bin/dhcp6/tests/callout_library_1.cc
index ffe05eba3f..768f374fe6 100644
--- a/src/bin/dhcp6/tests/callout_library_1.cc
+++ b/src/bin/dhcp6/tests/callout_library_1.cc
@@ -14,3 +14,17 @@ static const int LIBRARY_NUMBER = 1;
#include <config.h>
#include <dhcp6/tests/callout_library_common.h>
+
+// Functions accessed by the hooks framework use C linkage to avoid the name
+// mangling that accompanies use of the C++ compiler as well as to avoid
+// issues related to namespaces.
+extern "C" {
+
+/// @brief This function is called to retrieve the multi-threading compatibility.
+///
+/// @return 1 which means compatible with multi-threading.
+int multi_threading_compatible() {
+ return (1);
+}
+
+} // end extern "C"
diff --git a/src/bin/dhcp6/tests/callout_library_3.cc b/src/bin/dhcp6/tests/callout_library_3.cc
index 10bb0e6bc7..06a67ff804 100644
--- a/src/bin/dhcp6/tests/callout_library_3.cc
+++ b/src/bin/dhcp6/tests/callout_library_3.cc
@@ -17,6 +17,9 @@ static const int LIBRARY_NUMBER = 3;
using namespace isc::hooks;
+// Functions accessed by the hooks framework use C linkage to avoid the name
+// mangling that accompanies use of the C++ compiler as well as to avoid
+// issues related to namespaces.
extern "C" {
/// @brief Callout which appends library number and provided arguments to
@@ -43,4 +46,11 @@ dhcp6_srv_configured(CalloutHandle& handle) {
return (0);
}
+/// @brief This function is called to retrieve the multi-threading compatibility.
+///
+/// @return 0 which means not compatible with multi-threading.
+int multi_threading_compatible() {
+ return (0);
}
+
+} // end extern "C"
diff --git a/src/bin/dhcp6/tests/ctrl_dhcp6_srv_unittest.cc b/src/bin/dhcp6/tests/ctrl_dhcp6_srv_unittest.cc
index 3a9259e4a5..9cec48651c 100644
--- a/src/bin/dhcp6/tests/ctrl_dhcp6_srv_unittest.cc
+++ b/src/bin/dhcp6/tests/ctrl_dhcp6_srv_unittest.cc
@@ -18,10 +18,11 @@
#include <hooks/hooks_manager.h>
#include <log/logger_support.h>
#include <stats/stats_mgr.h>
+#include <util/boost_time_utils.h>
+#include <util/multi_threading_mgr.h>
#include <testutils/io_utils.h>
#include <testutils/unix_control_client.h>
#include <testutils/sandbox.h>
-#include <util/boost_time_utils.h>
#include "marker_file.h"
#include "test_libraries.h"
@@ -31,6 +32,7 @@
#include <iomanip>
#include <sstream>
+#include <thread>
#include <sys/select.h>
#include <sys/stat.h>
@@ -38,8 +40,6 @@
#include <cstdlib>
#include <unistd.h>
-#include <thread>
-
using namespace std;
using namespace isc;
using namespace isc::asiolink;
@@ -50,6 +50,7 @@ using namespace isc::dhcp::test;
using namespace isc::hooks;
using namespace isc::stats;
using namespace isc::test;
+using namespace isc::util;
namespace {
@@ -149,12 +150,14 @@ public:
socket_path_ = sandbox.join("/kea6.sock");
}
reset();
+ MultiThreadingMgr::instance().setMode(false);
}
/// @brief Destructor
~CtrlChannelDhcpv6SrvTest() {
server_.reset();
reset();
+ MultiThreadingMgr::instance().setMode(false);
};
/// @brief Returns pointer to the server's IO service.
@@ -472,6 +475,53 @@ TEST_F(CtrlChannelDhcpv6SrvTest, libreload) {
EXPECT_TRUE(checkMarkerFile(LOAD_MARKER_FILE, "1212"));
}
+// Check that the "libreload" command will reload libraries when MT is enabled
+TEST_F(CtrlChannelDhcpv6SrvTest, libreloadFailMultiThreading) {
+ createUnixChannelServer();
+
+ // Ensure no marker files to start with.
+ ASSERT_FALSE(checkMarkerFileExists(LOAD_MARKER_FILE));
+ ASSERT_FALSE(checkMarkerFileExists(UNLOAD_MARKER_FILE));
+
+ // Load two libraries
+ HookLibsCollection libraries;
+ libraries.push_back(make_pair(CALLOUT_LIBRARY_1, ConstElementPtr()));
+ libraries.push_back(make_pair(CALLOUT_LIBRARY_2, ConstElementPtr()));
+ HooksManager::loadLibraries(libraries);
+
+ // Check they are loaded.
+ HookLibsCollection loaded_libraries =
+ HooksManager::getLibraryInfo();
+ ASSERT_TRUE(libraries == loaded_libraries);
+
+ // ... which also included checking that the marker file created by the
+ // load functions exists and holds the correct value (of "12" - the
+ // first library appends "1" to the file, the second appends "2"). Also
+ // check that the unload marker file does not yet exist.
+ EXPECT_TRUE(checkMarkerFile(LOAD_MARKER_FILE, "12"));
+ EXPECT_FALSE(checkMarkerFileExists(UNLOAD_MARKER_FILE));
+
+ // Enable multi-threading before libreload command which should now fail
+ // as the second library is not multi-threading compatible.
+ MultiThreadingMgr::instance().setMode(true);
+
+ // Now execute the "libreload" command. This should cause the libraries
+ // to unload and to reload.
+ std::string response;
+ sendUnixCommand("{ \"command\": \"libreload\" }", response);
+ EXPECT_EQ("{ \"result\": 1, "
+ "\"text\": \"Failed to reload hooks libraries.\" }"
+ , response);
+
+ // Check that the libraries have unloaded and failed to reload. The
+ // libraries are unloaded in the reverse order to which they are loaded.
+ // When they load, they should append information to the loading marker
+ // file. Failing to load the second library will also unload the first
+ // library.
+ EXPECT_TRUE(checkMarkerFile(UNLOAD_MARKER_FILE, "211"));
+ EXPECT_TRUE(checkMarkerFile(LOAD_MARKER_FILE, "121"));
+}
+
typedef std::map<std::string, isc::data::ConstElementPtr> ElementMap;
// This test checks which commands are registered by the DHCPv6 server.
diff --git a/src/bin/dhcp6/tests/hooks_unittest.cc b/src/bin/dhcp6/tests/hooks_unittest.cc
index fb2d0b5aa5..985007734a 100644
--- a/src/bin/dhcp6/tests/hooks_unittest.cc
+++ b/src/bin/dhcp6/tests/hooks_unittest.cc
@@ -29,6 +29,7 @@
#include <cc/command_interpreter.h>
#include <dhcp6/tests/marker_file.h>
#include <dhcp6/tests/test_libraries.h>
+#include <util/multi_threading_mgr.h>
#include <boost/scoped_ptr.hpp>
#include <gtest/gtest.h>
@@ -967,12 +968,14 @@ public:
LoadUnloadDhcpv6SrvTest()
: Dhcpv6SrvTest() {
reset();
+ MultiThreadingMgr::instance().setMode(false);
}
/// @brief Destructor
~LoadUnloadDhcpv6SrvTest() {
server_.reset();
reset();
+ MultiThreadingMgr::instance().setMode(false);
};
/// @brief Reset hooks data
@@ -4874,8 +4877,8 @@ TEST_F(LoadUnloadDhcpv6SrvTest, unloadLibraries) {
libraries.push_back(make_pair(std::string(CALLOUT_LIBRARY_1),
ConstElementPtr()));
libraries.push_back(make_pair(std::string(CALLOUT_LIBRARY_2),
-
ConstElementPtr()));
+
ASSERT_TRUE(HooksManager::loadLibraries(libraries));
// Verify that they load functions created the LOAD_MARKER_FILE
@@ -4893,7 +4896,53 @@ TEST_F(LoadUnloadDhcpv6SrvTest, unloadLibraries) {
// this should be reflected in the unload file.
EXPECT_TRUE(checkMarkerFile(UNLOAD_MARKER_FILE, "21"));
EXPECT_TRUE(checkMarkerFile(LOAD_MARKER_FILE, "12"));
+}
+
+// Verifies that libraries incompatible with multi threading are not loaded by
+// the server
+// The callout libraries write their library index number to a marker
+// file upon load and unload, making it simple to test whether or not
+// the load and unload callouts have been invoked.
+TEST_F(LoadUnloadDhcpv6SrvTest, failLoadIncompatibleLibraries) {
+
+ ASSERT_NO_THROW(server_.reset(new NakedDhcpv6Srv(0)));
+
+ MultiThreadingMgr::instance().setMode(true);
+
+ // Ensure no marker files to start with.
+ ASSERT_FALSE(checkMarkerFileExists(LOAD_MARKER_FILE));
+ ASSERT_FALSE(checkMarkerFileExists(UNLOAD_MARKER_FILE));
+
+ // Load the test libraries
+ HookLibsCollection libraries;
+ libraries.push_back(make_pair(std::string(CALLOUT_LIBRARY_2),
+ ConstElementPtr()));
+
+ ASSERT_FALSE(HooksManager::loadLibraries(libraries));
+
+ // The library is missing multi_threading_compatible function so loading
+ // should fail
+ EXPECT_FALSE(checkMarkerFileExists(LOAD_MARKER_FILE));
+ EXPECT_FALSE(checkMarkerFileExists(UNLOAD_MARKER_FILE));
+ libraries.clear();
+ libraries.push_back(make_pair(std::string(CALLOUT_LIBRARY_3),
+ ConstElementPtr()));
+
+ ASSERT_FALSE(HooksManager::loadLibraries(libraries));
+
+ // The library is not multi threading compatible so loading should fail
+ EXPECT_FALSE(checkMarkerFileExists(LOAD_MARKER_FILE));
+ EXPECT_FALSE(checkMarkerFileExists(UNLOAD_MARKER_FILE));
+
+ // Destroy the server, instance which should unload the libraries.
+ server_.reset();
+
+ // Check that the libraries were unloaded. The libraries are
+ // unloaded in the reverse order to which they are loaded, and
+ // this should be reflected in the unload file.
+ EXPECT_FALSE(checkMarkerFileExists(LOAD_MARKER_FILE));
+ EXPECT_FALSE(checkMarkerFileExists(UNLOAD_MARKER_FILE));
}
// Checks if callouts installed on the dhcp6_srv_configured ared indeed called