| // Copyright 2018 The Chromium OS Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "shill/network/dhcp_controller.h" |
| |
| #include <arpa/inet.h> |
| #include <stdlib.h> |
| #include <sys/time.h> |
| #include <sys/wait.h> |
| |
| #include <optional> |
| |
| #include <base/check.h> |
| #include <base/files/file_util.h> |
| #include <base/logging.h> |
| #include <base/strings/stringprintf.h> |
| #include <base/time/time.h> |
| #include <chromeos/dbus/service_constants.h> |
| |
| #include "shill/control_interface.h" |
| #include "shill/event_dispatcher.h" |
| #include "shill/ipconfig.h" |
| #include "shill/logging.h" |
| #include "shill/metrics.h" |
| #include "shill/net/ip_address.h" |
| #include "shill/net/shill_time.h" |
| #include "shill/network/dhcp_provider.h" |
| #include "shill/network/dhcp_proxy_interface.h" |
| #include "shill/network/dhcpv4_config.h" |
| #include "shill/process_manager.h" |
| #include "shill/technology.h" |
| |
| namespace shill { |
| |
| namespace Logging { |
| static auto kModuleLogScope = ScopeLogger::kDHCP; |
| static std::string ObjectID(const DHCPController* d) { |
| if (d == nullptr) |
| return "(dhcp_controller)"; |
| else |
| return d->device_name(); |
| } |
| } // namespace Logging |
| |
| namespace { |
| |
| constexpr base::TimeDelta kAcquisitionTimeout = base::Seconds(30); |
| constexpr char kDHCPCDPath[] = "/sbin/dhcpcd"; |
| constexpr char kDHCPCDUser[] = "dhcp"; |
| constexpr char kDHCPCDGroup[] = "dhcp"; |
| constexpr char kDHCPCDPathFormatPID[] = "var/run/dhcpcd/dhcpcd-%s-4.pid"; |
| |
| } // namespace |
| |
| DHCPController::DHCPController(ControlInterface* control_interface, |
| EventDispatcher* dispatcher, |
| DHCPProvider* provider, |
| const std::string& device_name, |
| const std::string& lease_file_suffix, |
| bool arp_gateway, |
| const std::string& hostname, |
| Technology technology, |
| Metrics* metrics) |
| : control_interface_(control_interface), |
| provider_(provider), |
| device_name_(device_name), |
| lease_file_suffix_(lease_file_suffix), |
| technology_(technology), |
| pid_(0), |
| is_lease_active_(false), |
| arp_gateway_(arp_gateway), |
| is_gateway_arp_active_(false), |
| hostname_(hostname), |
| lease_acquisition_timeout_(kAcquisitionTimeout), |
| minimum_mtu_(IPConfig::kMinIPv4MTU), |
| root_("/"), |
| weak_ptr_factory_(this), |
| dispatcher_(dispatcher), |
| process_manager_(ProcessManager::GetInstance()), |
| metrics_(metrics), |
| time_(Time::GetInstance()) { |
| SLOG(this, 2) << __func__ << ": " << device_name; |
| if (lease_file_suffix_.empty()) { |
| lease_file_suffix_ = device_name; |
| } |
| } |
| |
| DHCPController::~DHCPController() { |
| SLOG(this, 2) << __func__ << ": " << device_name(); |
| |
| // Don't leave behind dhcpcd running. |
| Stop(__func__); |
| } |
| |
| void DHCPController::RegisterCallbacks(UpdateCallback update_callback, |
| FailureCallback failure_callback) { |
| update_callback_ = update_callback; |
| failure_callback_ = failure_callback; |
| } |
| |
| bool DHCPController::RequestIP() { |
| SLOG(this, 2) << __func__ << ": " << device_name(); |
| if (!pid_) { |
| return Start(); |
| } |
| if (!proxy_) { |
| LOG(ERROR) << "Unable to request IP before acquiring destination."; |
| return Restart(); |
| } |
| return RenewIP(); |
| } |
| |
| bool DHCPController::RenewIP() { |
| SLOG(this, 2) << __func__ << ": " << device_name(); |
| if (!pid_) { |
| return Start(); |
| } |
| if (!proxy_) { |
| LOG(ERROR) << "Unable to renew IP before acquiring destination."; |
| return false; |
| } |
| StopExpirationTimeout(); |
| proxy_->Rebind(device_name()); |
| StartAcquisitionTimeout(); |
| return true; |
| } |
| |
| bool DHCPController::ReleaseIP(ReleaseReason reason) { |
| SLOG(this, 2) << __func__ << ": " << device_name(); |
| if (!pid_) { |
| return true; |
| } |
| |
| // If we are using static IP and haven't retrieved a lease yet, we should |
| // allow the DHCP process to continue until we have a lease. |
| if (!is_lease_active_ && reason == kReleaseReasonStaticIP) { |
| return true; |
| } |
| |
| // If we are using gateway unicast ARP to speed up re-connect, don't |
| // give up our leases when we disconnect. |
| bool should_keep_lease = |
| reason == kReleaseReasonDisconnect && ShouldKeepLeaseOnDisconnect(); |
| |
| if (!should_keep_lease && proxy_.get()) { |
| proxy_->Release(device_name()); |
| } |
| Stop(__func__); |
| return true; |
| } |
| |
| void DHCPController::InitProxy(const std::string& service) { |
| if (!proxy_) { |
| LOG(INFO) << "Init DHCP Proxy: " << device_name() << " at " << service; |
| proxy_ = control_interface_->CreateDHCPProxy(service); |
| } |
| } |
| |
| void DHCPController::ProcessEventSignal(const std::string& reason, |
| const KeyValueStore& configuration) { |
| LOG(INFO) << "Event reason: " << reason; |
| if (reason == kReasonFail) { |
| LOG(ERROR) << "Received failure event from DHCP client."; |
| NotifyFailure(); |
| return; |
| } else if (reason == kReasonNak) { |
| // If we got a NAK, this means the DHCP server is active, and any |
| // Gateway ARP state we have is no longer sufficient. |
| LOG_IF(ERROR, is_gateway_arp_active_) |
| << "Received NAK event for our gateway-ARP lease."; |
| is_gateway_arp_active_ = false; |
| return; |
| } else if (reason != kReasonBound && reason != kReasonRebind && |
| reason != kReasonReboot && reason != kReasonRenew && |
| reason != kReasonGatewayArp) { |
| LOG(WARNING) << "Event ignored."; |
| return; |
| } |
| IPConfig::Properties properties; |
| CHECK(DHCPv4Config::ParseConfiguration(configuration, minimum_mtu_, |
| &properties)); |
| |
| // This needs to be set before calling OnIPConfigUpdated() below since |
| // those functions may indirectly call other methods like ReleaseIP that |
| // depend on or change this value. |
| set_is_lease_active(true); |
| |
| const bool is_gateway_arp = reason == kReasonGatewayArp; |
| // This is a non-authoritative confirmation that we or on the same |
| // network as the one we received a lease on previously. The DHCP |
| // client is still running, so we should not cancel the timeout |
| // until that completes. In the meantime, however, we can tentatively |
| // configure our network in anticipation of successful completion. |
| OnIPConfigUpdated(properties, /*new_lease_acquired=*/!is_gateway_arp); |
| is_gateway_arp_active_ = is_gateway_arp; |
| } |
| |
| std::optional<base::TimeDelta> DHCPController::TimeToLeaseExpiry() { |
| if (!current_lease_expiration_time_.has_value()) { |
| SLOG(this, 2) << __func__ << ": No current DHCP lease"; |
| return std::nullopt; |
| } |
| struct timeval now; |
| time_->GetTimeBoottime(&now); |
| if (now.tv_sec > current_lease_expiration_time_->tv_sec) { |
| SLOG(this, 2) << __func__ << ": Current DHCP lease has already expired"; |
| return std::nullopt; |
| } |
| return base::Seconds(current_lease_expiration_time_->tv_sec - now.tv_sec); |
| } |
| |
| void DHCPController::OnIPConfigUpdated(const IPConfig::Properties& properties, |
| bool new_lease_acquired) { |
| if (new_lease_acquired) { |
| StopAcquisitionTimeout(); |
| if (properties.lease_duration_seconds) { |
| UpdateLeaseExpirationTime(properties.lease_duration_seconds); |
| StartExpirationTimeout(base::Seconds(properties.lease_duration_seconds)); |
| } else { |
| LOG(WARNING) |
| << "Lease duration is zero; not starting an expiration timer."; |
| ResetLeaseExpirationTime(); |
| StopExpirationTimeout(); |
| } |
| } |
| |
| dispatcher_->PostTask( |
| FROM_HERE, base::BindOnce(&DHCPController::InvokeUpdateCallback, |
| weak_ptr_factory_.GetWeakPtr(), properties, |
| new_lease_acquired)); |
| } |
| |
| void DHCPController::NotifyFailure() { |
| StopAcquisitionTimeout(); |
| StopExpirationTimeout(); |
| |
| dispatcher_->PostTask(FROM_HERE, |
| base::BindOnce(&DHCPController::InvokeFailureCallback, |
| weak_ptr_factory_.GetWeakPtr())); |
| } |
| |
| bool DHCPController::IsEphemeralLease() const { |
| return lease_file_suffix_ == device_name(); |
| } |
| |
| bool DHCPController::Start() { |
| SLOG(this, 2) << __func__ << ": " << device_name(); |
| |
| // Setup program arguments. |
| auto args = GetFlags(); |
| std::string interface_arg(device_name()); |
| if (lease_file_suffix_ != device_name()) { |
| interface_arg = base::StringPrintf("%s=%s", device_name().c_str(), |
| lease_file_suffix_.c_str()); |
| } |
| args.push_back(interface_arg); |
| |
| ProcessManager::MinijailOptions minijail_options; |
| minijail_options.user = kDHCPCDUser; |
| minijail_options.group = kDHCPCDGroup; |
| minijail_options.capmask = |
| CAP_TO_MASK(CAP_NET_BIND_SERVICE) | CAP_TO_MASK(CAP_NET_BROADCAST) | |
| CAP_TO_MASK(CAP_NET_ADMIN) | CAP_TO_MASK(CAP_NET_RAW); |
| minijail_options.inherit_supplementary_groups = false; |
| // TODO(crrev.com/c/3162356): Check if |close_nonstd_fds| can be set to true. |
| minijail_options.close_nonstd_fds = false; |
| pid_t pid = process_manager_->StartProcessInMinijail( |
| FROM_HERE, base::FilePath(kDHCPCDPath), args, {}, minijail_options, |
| base::BindOnce(&DHCPController::OnProcessExited, |
| weak_ptr_factory_.GetWeakPtr())); |
| if (pid < 0) { |
| return false; |
| } |
| pid_ = pid; |
| LOG(INFO) << "Spawned " << kDHCPCDPath << " with pid: " << pid_; |
| provider_->BindPID(pid_, weak_ptr_factory_.GetWeakPtr()); |
| StartAcquisitionTimeout(); |
| return true; |
| } |
| |
| void DHCPController::Stop(const char* reason) { |
| LOG_IF(INFO, pid_) << "Stopping " << pid_ << " (" << reason << ")"; |
| KillClient(); |
| // KillClient waits for the client to terminate so it's safe to cleanup the |
| // state. |
| CleanupClientState(); |
| } |
| |
| void DHCPController::KillClient() { |
| if (!pid_) { |
| return; |
| } |
| |
| // Pass the termination responsibility to ProcessManager. |
| // ProcessManager will try to terminate the process using SIGTERM, then |
| // SIGKill signals. It will log an error message if it is not able to |
| // terminate the process in a timely manner. |
| process_manager_->StopProcessAndBlock(pid_); |
| } |
| |
| bool DHCPController::Restart() { |
| Stop(__func__); |
| return Start(); |
| } |
| |
| void DHCPController::OnProcessExited(int exit_status) { |
| CHECK(pid_); |
| if (exit_status == EXIT_SUCCESS) { |
| SLOG(nullptr, 2) << "pid " << pid_ << " exit status " << exit_status; |
| } else { |
| LOG(WARNING) << "pid " << pid_ << " exit status " << exit_status; |
| } |
| CleanupClientState(); |
| } |
| |
| void DHCPController::CleanupClientState() { |
| SLOG(this, 2) << __func__ << ": " << device_name(); |
| StopAcquisitionTimeout(); |
| StopExpirationTimeout(); |
| |
| proxy_.reset(); |
| if (pid_) { |
| int pid = pid_; |
| pid_ = 0; |
| provider_->UnbindPID(pid); |
| } |
| is_lease_active_ = false; |
| |
| // Delete lease file if it is ephemeral. |
| if (IsEphemeralLease()) { |
| base::DeleteFile(root().Append(base::StringPrintf( |
| DHCPProvider::kDHCPCDPathFormatLease, device_name().c_str()))); |
| } |
| base::DeleteFile(root().Append( |
| base::StringPrintf(kDHCPCDPathFormatPID, device_name().c_str()))); |
| is_gateway_arp_active_ = false; |
| } |
| |
| bool DHCPController::ShouldFailOnAcquisitionTimeout() const { |
| // Continue to use previous lease if gateway ARP is active. |
| return !is_gateway_arp_active_; |
| } |
| |
| // Return true if we should keep the lease on disconnect. |
| bool DHCPController::ShouldKeepLeaseOnDisconnect() const { |
| // If we are using gateway unicast ARP to speed up re-connect, don't |
| // give up our leases when we disconnect. |
| return arp_gateway_; |
| } |
| |
| std::vector<std::string> DHCPController::GetFlags() { |
| std::vector<std::string> flags; |
| flags.push_back("-B"); // Run in foreground. |
| flags.push_back("-q"); // Only warnings+errors to stderr. |
| flags.push_back("-4"); // IPv4 only. |
| |
| // Apply options from DhcpProperties when applicable. |
| if (!hostname_.empty()) { |
| flags.push_back("-h"); // Request hostname from server |
| flags.push_back(hostname_); |
| } |
| |
| if (arp_gateway_) { |
| flags.push_back("-R"); // ARP for default gateway. |
| flags.push_back("--unicast"); // Enable unicast ARP on renew. |
| } |
| |
| return flags; |
| } |
| |
| void DHCPController::StartAcquisitionTimeout() { |
| CHECK(lease_expiration_callback_.IsCancelled()); |
| lease_acquisition_timeout_callback_.Reset( |
| Bind(&DHCPController::ProcessAcquisitionTimeout, |
| weak_ptr_factory_.GetWeakPtr())); |
| dispatcher_->PostDelayedTask(FROM_HERE, |
| lease_acquisition_timeout_callback_.callback(), |
| lease_acquisition_timeout_); |
| } |
| |
| void DHCPController::StopAcquisitionTimeout() { |
| lease_acquisition_timeout_callback_.Cancel(); |
| } |
| |
| void DHCPController::ProcessAcquisitionTimeout() { |
| LOG(ERROR) << "Timed out waiting for DHCP lease on " << device_name() << " " |
| << "(after " << lease_acquisition_timeout_.InSeconds() |
| << " seconds)."; |
| if (!ShouldFailOnAcquisitionTimeout()) { |
| LOG(INFO) << "Continuing to use our previous lease, due to gateway-ARP."; |
| } else { |
| NotifyFailure(); |
| } |
| } |
| |
| void DHCPController::StartExpirationTimeout(base::TimeDelta lease_duration) { |
| CHECK(lease_acquisition_timeout_callback_.IsCancelled()); |
| SLOG(this, 2) << __func__ << ": " << device_name() << ": " |
| << "Lease timeout is " << lease_duration.InSeconds() |
| << " seconds."; |
| lease_expiration_callback_.Reset( |
| BindOnce(&DHCPController::ProcessExpirationTimeout, |
| weak_ptr_factory_.GetWeakPtr(), lease_duration)); |
| dispatcher_->PostDelayedTask(FROM_HERE, lease_expiration_callback_.callback(), |
| lease_duration); |
| } |
| |
| void DHCPController::StopExpirationTimeout() { |
| lease_expiration_callback_.Cancel(); |
| } |
| |
| void DHCPController::ProcessExpirationTimeout(base::TimeDelta lease_duration) { |
| LOG(ERROR) << "DHCP lease expired on " << device_name() |
| << "; restarting DHCP client instance."; |
| |
| metrics_->SendToUMA( |
| metrics_->GetFullMetricName( |
| Metrics::kMetricExpiredLeaseLengthSecondsSuffix, technology_), |
| lease_duration.InSeconds(), Metrics::kMetricExpiredLeaseLengthSecondsMin, |
| Metrics::kMetricExpiredLeaseLengthSecondsMax, |
| Metrics::kMetricExpiredLeaseLengthSecondsNumBuckets); |
| |
| if (!Restart()) { |
| NotifyFailure(); |
| } |
| } |
| |
| void DHCPController::UpdateLeaseExpirationTime(uint32_t new_lease_duration) { |
| struct timeval new_expiration_time; |
| time_->GetTimeBoottime(&new_expiration_time); |
| new_expiration_time.tv_sec += new_lease_duration; |
| current_lease_expiration_time_ = new_expiration_time; |
| } |
| |
| void DHCPController::ResetLeaseExpirationTime() { |
| current_lease_expiration_time_ = std::nullopt; |
| } |
| |
| void DHCPController::InvokeUpdateCallback(const IPConfig::Properties properties, |
| bool new_lease_acquired) { |
| if (!update_callback_.is_null()) { |
| update_callback_.Run(properties, new_lease_acquired); |
| } |
| } |
| |
| void DHCPController::InvokeFailureCallback() { |
| if (!failure_callback_.is_null()) { |
| failure_callback_.Run(); |
| } |
| } |
| |
| } // namespace shill |