// // Copyright (C) 2012 The Android Open Source Project // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. // #ifndef SHILL_DHCP_DHCP_PROVIDER_H_ #define SHILL_DHCP_DHCP_PROVIDER_H_ #include <map> #include <memory> #include <set> #include <string> #include <base/files/file_path.h> #include <base/lazy_instance.h> #include <gtest/gtest_prod.h> // for FRIEND_TEST #include "shill/dhcp_properties.h" #include "shill/refptr_types.h" namespace shill { class ControlInterface; class DHCPCDListenerInterface; class EventDispatcher; class Metrics; // DHCPProvider is a singleton providing the main DHCP configuration // entrypoint. Once the provider is initialized through its Init method, DHCP // configurations for devices can be obtained through its CreateConfig // method. For example, a single DHCP configuration request can be initiated as: // // DHCPProvider::GetInstance()->CreateIPv4Config(device_name, // lease_file_suffix, // arp_gateway, // dhcp_props)->Request(); class DHCPProvider { public: static constexpr char kDHCPCDPathFormatLease[] = "var/lib/dhcpcd/dhcpcd-%s.lease"; #ifndef DISABLE_DHCPV6 static constexpr char kDHCPCDPathFormatLease6[] = "var/lib/dhcpcd/dhcpcd-%s.lease6"; #endif // DISABLE_DHCPV6 virtual ~DHCPProvider(); // This is a singleton -- use DHCPProvider::GetInstance()->Foo(). static DHCPProvider* GetInstance(); // Initializes the provider singleton. This method hooks up a D-Bus signal // listener that catches signals from spawned DHCP clients and dispatches them // to the appropriate DHCP configuration instance. virtual void Init(ControlInterface* control_interface, EventDispatcher* dispatcher, Metrics* metrics); // Called on shutdown to release |listener_|. void Stop(); // Creates a new DHCPv4Config for |device_name|. The DHCP configuration for // the device can then be initiated through DHCPConfig::Request and // DHCPConfig::Renew. If |host_name| is not-empty, it is placed in the DHCP // request to allow the server to map the request to a specific user-named // origin. The DHCP lease file will contain the suffix supplied // in |lease_file_suffix| if non-empty, otherwise |device_name|. If // |arp_gateway| is true, the DHCP client will ARP for the gateway IP // address as an additional safeguard against the issued IP address being // in-use by another station. virtual DHCPConfigRefPtr CreateIPv4Config( const std::string& device_name, const std::string& lease_file_suffix, bool arp_gateway, const DhcpProperties& dhcp_props); #ifndef DISABLE_DHCPV6 // Create a new DHCPv6Config for |device_name|. virtual DHCPConfigRefPtr CreateIPv6Config( const std::string& device_name, const std::string& lease_file_suffix); #endif // Returns the DHCP configuration associated with DHCP client |pid|. Return // nullptr if |pid| is not bound to a configuration. DHCPConfigRefPtr GetConfig(int pid); // Binds a |pid| to a DHCP |config|. When a DHCP config spawns a new DHCP // client, it binds itself to that client's |pid|. virtual void BindPID(int pid, const DHCPConfigRefPtr& config); // Unbinds a |pid|. This method is used by a DHCP config to signal the // provider that the DHCP client has been terminated. This may result in // destruction of the DHCP config instance if its reference count goes to 0. virtual void UnbindPID(int pid); // Destroy lease file associated with this |name|. virtual void DestroyLease(const std::string& name); // Returns true if |pid| was recently unbound from the provider. bool IsRecentlyUnbound(int pid); protected: DHCPProvider(); private: friend struct base::DefaultLazyInstanceTraits<DHCPProvider>; friend class CellularTest; friend class DHCPProviderTest; friend class DeviceInfoTest; friend class DeviceTest; FRIEND_TEST(DHCPProviderTest, CreateIPv4Config); FRIEND_TEST(DHCPProviderTest, DestroyLease); typedef std::map<int, DHCPConfigRefPtr> PIDConfigMap; // Retire |pid| from the set of recently retired PIDs. void RetireUnboundPID(int pid); // A single listener is used to catch signals from all DHCP clients and // dispatch them to the appropriate DHCP configuration instance. std::unique_ptr<DHCPCDListenerInterface> listener_; // A map that binds PIDs to DHCP configuration instances. PIDConfigMap configs_; base::FilePath root_; ControlInterface* control_interface_; EventDispatcher* dispatcher_; Metrics* metrics_; // Track the set of PIDs recently unbound from the provider in case messages // arrive addressed from them. std::set<int> recently_unbound_pids_; DISALLOW_COPY_AND_ASSIGN(DHCPProvider); }; } // namespace shill #endif // SHILL_DHCP_DHCP_PROVIDER_H_