ctrl_dhcp6_srv.cc 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162
  1. // Copyright (C) 2012 Internet Systems Consortium, Inc. ("ISC")
  2. //
  3. // Permission to use, copy, modify, and/or distribute this software for any
  4. // purpose with or without fee is hereby granted, provided that the above
  5. // copyright notice and this permission notice appear in all copies.
  6. //
  7. // THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
  8. // REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
  9. // AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
  10. // INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
  11. // LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
  12. // OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
  13. // PERFORMANCE OF THIS SOFTWARE.
  14. #include <config.h>
  15. #include <cassert>
  16. #include <iostream>
  17. #include <asiolink/asiolink.h>
  18. #include <cc/data.h>
  19. #include <cc/session.h>
  20. #include <cc/session.h>
  21. #include <config/ccsession.h>
  22. #include <dhcp6/ctrl_dhcp6_srv.h>
  23. #include <dhcp6/dhcp6_log.h>
  24. #include <dhcp6/spec_config.h>
  25. #include <dhcp/iface_mgr.h>
  26. #include <exceptions/exceptions.h>
  27. #include <util/buffer.h>
  28. using namespace isc::asiolink;
  29. using namespace isc::cc;
  30. using namespace isc::config;
  31. using namespace isc::data;
  32. using namespace isc::dhcp;
  33. using namespace isc::log;
  34. using namespace isc::util;
  35. using namespace std;
  36. namespace isc {
  37. namespace dhcp {
  38. ControlledDhcpv6Srv* ControlledDhcpv6Srv::server_ = NULL;
  39. ConstElementPtr
  40. ControlledDhcpv6Srv::dhcp6ConfigHandler(ConstElementPtr new_config) {
  41. LOG_DEBUG(dhcp6_logger, DBG_DHCP6_COMMAND, DHCP6_CONFIG_UPDATE)
  42. .arg(new_config->str());
  43. ConstElementPtr answer = isc::config::createAnswer(0,
  44. "Thank you for sending config.");
  45. return (answer);
  46. }
  47. ConstElementPtr
  48. ControlledDhcpv6Srv::dhcp6CommandHandler(const string& command, ConstElementPtr args) {
  49. LOG_DEBUG(dhcp6_logger, DBG_DHCP6_COMMAND, DHCP6_COMMAND_RECEIVED)
  50. .arg(command).arg(args->str());
  51. if (command == "shutdown") {
  52. if (ControlledDhcpv6Srv::server_) {
  53. ControlledDhcpv6Srv::server_->shutdown();
  54. } else {
  55. LOG_WARN(dhcp6_logger, DHCP6_NOT_RUNNING);
  56. ConstElementPtr answer = isc::config::createAnswer(1,
  57. "Shutdown failure.");
  58. return (answer);
  59. }
  60. ConstElementPtr answer = isc::config::createAnswer(0,
  61. "Shutting down.");
  62. return (answer);
  63. }
  64. ConstElementPtr answer = isc::config::createAnswer(1,
  65. "Unrecognized command.");
  66. return (answer);
  67. }
  68. void ControlledDhcpv6Srv::sessionReader(void) {
  69. // Process one asio event. If there are more events, iface_mgr will call
  70. // this callback more than once.
  71. if (server_) {
  72. server_->io_service_.run_one();
  73. }
  74. }
  75. void ControlledDhcpv6Srv::establishSession() {
  76. string specfile;
  77. if (getenv("B10_FROM_BUILD")) {
  78. specfile = string(getenv("B10_FROM_BUILD")) +
  79. "/src/bin/dhcp6/dhcp6.spec";
  80. } else {
  81. specfile = string(DHCP6_SPECFILE_LOCATION);
  82. }
  83. /// @todo: Check if session is not established already. Throw, if it is.
  84. LOG_DEBUG(dhcp6_logger, DBG_DHCP6_START, DHCP6_CCSESSION_STARTING)
  85. .arg(specfile);
  86. cc_session_ = new Session(io_service_.get_io_service());
  87. config_session_ = new ModuleCCSession(specfile, *cc_session_,
  88. dhcp6ConfigHandler,
  89. dhcp6CommandHandler, false);
  90. config_session_->start();
  91. /// Integrate the asynchronous I/O model of BIND 10 configuration
  92. /// control with the "select" model of the DHCP server. This is
  93. /// fully explained in \ref dhcpv6Session.
  94. int ctrl_socket = cc_session_->getSocketDesc();
  95. LOG_DEBUG(dhcp6_logger, DBG_DHCP6_START, DHCP6_CCSESSION_STARTED)
  96. .arg(ctrl_socket);
  97. IfaceMgr::instance().set_session_socket(ctrl_socket, sessionReader);
  98. }
  99. void ControlledDhcpv6Srv::disconnectSession() {
  100. if (config_session_) {
  101. delete config_session_;
  102. config_session_ = NULL;
  103. }
  104. if (cc_session_) {
  105. cc_session_->disconnect();
  106. delete cc_session_;
  107. cc_session_ = NULL;
  108. }
  109. // deregister session socket
  110. IfaceMgr::instance().set_session_socket(IfaceMgr::INVALID_SOCKET, NULL);
  111. }
  112. ControlledDhcpv6Srv::ControlledDhcpv6Srv(uint16_t port /*= DHCP6_SERVER_PORT*/)
  113. :Dhcpv6Srv(port), cc_session_(NULL), config_session_(NULL) {
  114. server_ = this; // remember this instance for use in callback
  115. }
  116. void ControlledDhcpv6Srv::shutdown() {
  117. io_service_.stop(); // Stop ASIO transmissions
  118. Dhcpv6Srv::shutdown(); // Initiate DHCPv6 shutdown procedure.
  119. }
  120. ControlledDhcpv6Srv::~ControlledDhcpv6Srv() {
  121. disconnectSession();
  122. server_ = NULL; // forget this instance. There should be no callback anymore
  123. // at this stage anyway.
  124. }
  125. isc::data::ConstElementPtr
  126. ControlledDhcpv6Srv::execDhcpv6ServerCommand(const std::string& command_id,
  127. isc::data::ConstElementPtr args) {
  128. try {
  129. return (dhcp6CommandHandler(command_id, args));
  130. } catch (const Exception& ex) {
  131. ConstElementPtr answer = isc::config::createAnswer(1, ex.what());
  132. return (answer);
  133. }
  134. }
  135. };
  136. };