// Copyright 2012 The Chromium Authors // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef NET_BASE_ADDRESS_TRACKER_LINUX_H_ #define NET_BASE_ADDRESS_TRACKER_LINUX_H_ #include // Needed to include netlink. // Mask superfluous definition of |struct net|. This is fixed in Linux 2.6.38. #define net net_kernel #include #undef net #include #include #include #include #include #include "base/compiler_specific.h" #include "base/files/file_descriptor_watcher_posix.h" #include "base/files/scoped_file.h" #include "base/functional/callback.h" #include "base/gtest_prod_util.h" #include "base/memory/raw_ref.h" #include "base/sequence_checker.h" #include "base/synchronization/condition_variable.h" #include "base/synchronization/lock.h" #include "base/task/sequenced_task_runner.h" #include "base/thread_annotations.h" #include "net/base/address_map_linux.h" #include "net/base/ip_address.h" #include "net/base/net_export.h" #include "net/base/network_change_notifier.h" namespace net::test { class AddressTrackerLinuxTest; } namespace net::internal { // Keeps track of network interface addresses using rtnetlink. Used by // NetworkChangeNotifier to provide signals to registered IPAddressObservers. // // In tracking mode, this class should mostly be used on a single sequence, // except GetAddressMap() and GetOnlineLinks() (AddressMapOwnerLinux overrides) // which can be called on any thread. The main sequence should be able to block // (e.g. use a base::SequencedTaskRunner with base::MayBlock()). // // In non-tracking mode this should be used on a single thread. class NET_EXPORT_PRIVATE AddressTrackerLinux : public AddressMapOwnerLinux { public: // Non-tracking version constructor: it takes a snapshot of the // current system configuration. Once Init() returns, the // configuration is available through GetOnlineLinks() and // GetAddressMap(). AddressTrackerLinux(); // Tracking version constructor: it will run |address_callback| when // the AddressMap changes, |link_callback| when the list of online // links changes, and |tunnel_callback| when the list of online // tunnels changes. // |ignored_interfaces| is the list of interfaces to ignore. Changes to an // ignored interface will not cause any callback to be run. An ignored // interface will not have entries in GetAddressMap() and GetOnlineLinks(). // NOTE: Only ignore interfaces not used to connect to the internet. Adding // interfaces used to connect to the internet can cause critical network // changed signals to be lost allowing incorrect stale state to persist. // // |blocking_thread_runner| is the sequence on which this AddressTrackerLinux // will run. The AddressTrackerLinux can block in tracking mode and so it // should run on a sequence that can block, e.g. a base::SequencedTaskRunner // with base::MayBlock(). If nullptr, SetDiffCallback() cannot be used off of // the AddressTrackerLinux's sequence. AddressTrackerLinux(const base::RepeatingClosure& address_callback, const base::RepeatingClosure& link_callback, const base::RepeatingClosure& tunnel_callback, const std::unordered_set& ignored_interfaces, scoped_refptr blocking_thread_runner = nullptr); ~AddressTrackerLinux() override; // In tracking mode, it starts watching the system configuration for // changes. The current thread must have a MessageLoopForIO. In // non-tracking mode, once Init() returns, a snapshot of the system // configuration is available through GetOnlineLinks() and // GetAddressMap(). void Init(); // Same as Init(), except instead of creating and binding a netlink socket, // this AddressTrackerLinux will send and receive messages from |fd|. void InitWithFdForTesting(base::ScopedFD fd); // AddressMapOwnerLinux implementation (callable on any thread): AddressMap GetAddressMap() const override; // Returns set of interface indices for online interfaces. std::unordered_set GetOnlineLinks() const override; AddressTrackerLinux* GetAddressTrackerLinux() override; // This returns the current AddressMap and set of online links, and atomically // starts recording diffs to those structures. This can be called on any // thread, and must be called called before SetDiffCallback() below. Available // only in tracking mode. std::pair> GetInitialDataAndStartRecordingDiffs(); // Called after GetInitialDataAndStartRecordingDiffs(). // // Whenever the AddressMap or the set of online links (returned by the above // two methods) changes, this callback is called on AddressTrackerLinux's // sequence. On the first call, |diff_callback| is called synchronously with // the set of diffs that have been built since // GetInitialDataAndStartRecordingDiffs() was called. If there are none, // |diff_callback| won't be called. // // This is only available in tracking mode. It can be called on any thread, // but it will post a task to the AddressTrackerLinux's sequence and therefore // will finish asynchronously. The caller MUST ENSURE that the // AddressTrackerLinux is not deleted until this task finishes. // This also requires |sequenced_task_runner_| to be set by the // AddressTrackerLinux constructor above. // // Note that other threads may see updated AddressMaps by calling // GetAddressMap() before |diff_callback| is ever called. void SetDiffCallback(DiffCallback diff_callback); // Implementation of NetworkChangeNotifierLinux::GetCurrentConnectionType(). // Safe to call from any thread, but will block until Init() has completed. NetworkChangeNotifier::ConnectionType GetCurrentConnectionType(); // Returns the name for the interface with interface index |interface_index|. // |buf| should be a pointer to an array of size IFNAMSIZ. The returned // pointer will point to |buf|. This function acts like if_indextoname which // cannot be used as net/if.h cannot be mixed with linux/if.h. We'll stick // with exclusively talking to the kernel and not the C library. static char* GetInterfaceName(int interface_index, char* buf); // Does |name| refer to a tunnel interface? static bool IsTunnelInterfaceName(const char* name); private: friend class net::test::AddressTrackerLinuxTest; FRIEND_TEST_ALL_PREFIXES(AddressTrackerLinuxNetlinkTest, TestInitializeTwoTrackers); FRIEND_TEST_ALL_PREFIXES(AddressTrackerLinuxNetlinkTest, TestInitializeTwoTrackersInPidNamespaces); friend int ChildProcessInitializeTrackerForTesting(); // In tracking mode, holds |lock| while alive. In non-tracking mode, // enforces single-threaded access. class SCOPED_LOCKABLE AddressTrackerAutoLock { public: AddressTrackerAutoLock(const AddressTrackerLinux& tracker, base::Lock& lock) EXCLUSIVE_LOCK_FUNCTION(lock); AddressTrackerAutoLock(const AddressTrackerAutoLock&) = delete; AddressTrackerAutoLock& operator=(const AddressTrackerAutoLock&) = delete; ~AddressTrackerAutoLock() UNLOCK_FUNCTION(); private: const raw_ref tracker_; const raw_ref lock_; }; // A function that returns the name of an interface given the interface index // in |interface_index|. |ifname| should be a buffer of size IFNAMSIZ. The // function should return a pointer to |ifname|. typedef char* (*GetInterfaceNameFunction)(int interface_index, char* ifname); // Retrieves a dump of the current AddressMap and set of online links as part // of initialization. Expects |netlink_fd_| to exist already. void DumpInitialAddressesAndWatch(); // Sets |*address_changed| to indicate whether |address_map_| changed and // sets |*link_changed| to indicate if |online_links_| changed and sets // |*tunnel_changed| to indicate if |online_links_| changed with regards to a // tunnel interface while reading messages from |netlink_fd_|. // // If |address_map_| has changed and |address_map_diff_| is not nullopt, // |*address_map_diff_| is populated with the changes to the AddressMap. // Similarly, if |online_links_| has changed and |online_links_diff_| is not // nullopt, |*online_links_diff| is populated with the changes to the set of // online links. void ReadMessages(bool* address_changed, bool* link_changed, bool* tunnel_changed); // Sets |*address_changed| to true if |address_map_| changed, sets // |*link_changed| to true if |online_links_| changed, sets |*tunnel_changed| // to true if |online_links_| changed with regards to a tunnel interface while // reading the message from |buffer|. // // If |address_map_| has changed and |address_map_diff_| is not nullopt, // |*address_map_diff_| is populated with the changes to the AddressMap. // Similarly, if |online_links_| has changed and |online_links_diff_| is not // nullopt, |*online_links_diff| is populated with the changes to the set of // online links. void HandleMessage(const char* buffer, int length, bool* address_changed, bool* link_changed, bool* tunnel_changed); // Call when some part of initialization failed; forces online and unblocks. void AbortAndForceOnline(); // Called by |watcher_| when |netlink_fd_| can be read without blocking. void OnFileCanReadWithoutBlocking(); // Does |interface_index| refer to a tunnel interface? bool IsTunnelInterface(int interface_index) const; // Is interface with index |interface_index| in list of ignored interfaces? bool IsInterfaceIgnored(int interface_index) const; // Updates current_connection_type_ based on the network list. void UpdateCurrentConnectionType(); // Passes |address_map_diff_| and |online_links_diff_| to |diff_callback_| as // arguments, and then clears them. void RunDiffCallback(); // Used by AddressTrackerLinuxTest, returns the number of threads waiting // for |connection_type_initialized_cv_|. int GetThreadsWaitingForConnectionTypeInitForTesting(); // Used by AddressTrackerLinuxNetlinkTest, returns true iff `Init` succeeded. // Undefined for non-tracking mode. bool DidTrackingInitSucceedForTesting() const; AddressMapDiff& address_map_diff_for_testing() { AddressTrackerAutoLock lock(*this, address_map_lock_); return address_map_diff_.value(); } OnlineLinksDiff& online_links_diff_for_testing() { AddressTrackerAutoLock lock(*this, online_links_lock_); return online_links_diff_.value(); } // Gets the name of an interface given the interface index |interface_index|. // May return empty string if it fails but should not return NULL. This is // overridden by tests. GetInterfaceNameFunction get_interface_name_; DiffCallback diff_callback_ GUARDED_BY_CONTEXT(sequence_checker_); base::RepeatingClosure address_callback_ GUARDED_BY_CONTEXT(sequence_checker_); base::RepeatingClosure link_callback_ GUARDED_BY_CONTEXT(sequence_checker_); base::RepeatingClosure tunnel_callback_ GUARDED_BY_CONTEXT(sequence_checker_); // Note that |watcher_| must be inactive when |netlink_fd_| is closed. base::ScopedFD netlink_fd_ GUARDED_BY_CONTEXT(sequence_checker_); std::unique_ptr watcher_ GUARDED_BY_CONTEXT(sequence_checker_); mutable base::Lock address_map_lock_; AddressMap address_map_ GUARDED_BY(address_map_lock_); std::optional address_map_diff_; // Set of interface indices for links that are currently online. mutable base::Lock online_links_lock_ ACQUIRED_AFTER(address_map_lock_); std::unordered_set online_links_ GUARDED_BY(online_links_lock_); std::optional online_links_diff_; // Set of interface names that should be ignored. const std::unordered_set ignored_interfaces_; base::Lock connection_type_lock_; bool connection_type_initialized_ GUARDED_BY(connection_type_lock_) = false; base::ConditionVariable connection_type_initialized_cv_; NetworkChangeNotifier::ConnectionType current_connection_type_ GUARDED_BY( connection_type_lock_) = NetworkChangeNotifier::CONNECTION_NONE; int threads_waiting_for_connection_type_initialization_ GUARDED_BY(connection_type_lock_) = 0; const bool tracking_; // This can be set by the tracking constructor. scoped_refptr sequenced_task_runner_; // This SequenceChecker is still useful so instance variables above can be // marked GUARDED_BY_CONTEXT(sequence_checker_). SEQUENCE_CHECKER(sequence_checker_); base::WeakPtrFactory weak_ptr_factory_{this}; }; } // namespace net::internal #endif // NET_BASE_ADDRESS_TRACKER_LINUX_H_