// // Copyright (C) 2015 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_ICMP_SESSION_H_ #define SHILL_ICMP_SESSION_H_ #if defined(__ANDROID__) #include <linux/icmp.h> #else #include <netinet/ip_icmp.h> #endif // __ANDROID__ #include <map> #include <memory> #include <set> #include <string> #include <utility> #include <vector> #include <base/callback.h> #include <base/cancelable_callback.h> #include <base/macros.h> #include <base/memory/weak_ptr.h> #include <base/time/default_tick_clock.h> #include <base/time/tick_clock.h> #include <gtest/gtest_prod.h> // for FRIEND_TEST #include "shill/icmp.h" #include "shill/net/io_handler.h" namespace shill { class EventDispatcher; class IPAddress; // The IcmpSession class encapsulates the task of performing a stateful exchange // of echo requests and echo replies between this host and another (i.e. ping). // The Icmp class is used to perform the sending of echo requests. Each // IcmpSession object only allows one ICMP session to be running at one time. // Multiple ICMP sessions can be run concurrently by creating multiple // IcmpSession objects. class IcmpSession { public: // The result of an ICMP session is a vector of time deltas representing how // long it took to receive a echo reply for each sent echo request. The vector // is sorted in the order that the echo requests were sent. Zero time deltas // represent echo requests that we did not receive a corresponding reply for. using IcmpSessionResult = std::vector<base::TimeDelta>; using IcmpSessionResultCallback = base::Callback<void(const IcmpSessionResult&)>; explicit IcmpSession(EventDispatcher* dispatcher); // We always call IcmpSession::Stop in the destructor to clean up, in case an // ICMP session is still in progress. virtual ~IcmpSession(); // Starts an ICMP session, sending |kNumEchoRequestsToSend| echo requests to // |destination|, |kEchoRequestIntervalSeconds| apart. |result_callback| will // be called a) after all echo requests are sent and all echo replies are // received, or b) after |kTimeoutSeconds| have passed. |result_callback| will // only be invoked once on the first occurrence of either of these events. virtual bool Start(const IPAddress& destination, const IcmpSessionResultCallback& result_callback); // Stops the current ICMP session by closing the ICMP socket and resetting // callbacks. Does nothing if a ICMP session is not started. virtual void Stop(); bool IsStarted() { return icmp_->IsStarted(); } // Utility function that returns false iff |result| indicates that no echo // replies were received to any ICMP echo request that was sent during the // ICMP session that generated |result|. static bool AnyRepliesReceived(const IcmpSessionResult& result); // Utility function that returns the packet loss rate for the ICMP session // that generated |result| is greater than |percentage_threshold| percent. // The percentage packet loss determined by this function will be rounded // down to the closest integer percentage value. |percentage_threshold| is // expected to be a non-negative integer value. static bool IsPacketLossPercentageGreaterThan(const IcmpSessionResult& result, int percentage_threshold); private: using SentRecvTimePair = std::pair<base::TimeTicks, base::TimeTicks>; friend class IcmpSessionTest; FRIEND_TEST(IcmpSessionTest, Constructor); // for |echo_id_| static uint16_t kNextUniqueEchoId; // unique across IcmpSession objects static const int kTotalNumEchoRequests; static const int kEchoRequestIntervalSeconds; static const size_t kTimeoutSeconds; // Sends a single echo request to |destination|. This function will call // itself repeatedly via the event loop every |kEchoRequestIntervalSeconds| // until |kNumEchoRequestToSend| echo requests are sent or the timeout is // reached. void TransmitEchoRequestTask(const IPAddress& destination); // Called when an ICMP packet is received. void OnEchoReplyReceived(InputData* data); // Helper function that generates the result of the current ICMP session. IcmpSessionResult GenerateIcmpResult(); // Called when the input handler |echo_reply_handler_| encounters an error. void OnEchoReplyError(const std::string& error_msg); // Calls |result_callback_| with the results collected so far, then stops the // IcmpSession. This function is called when the ICMP session successfully // completes, or when it times out. Does nothing if an ICMP session is not // started. void ReportResultAndStopSession(); base::WeakPtrFactory<IcmpSession> weak_ptr_factory_; EventDispatcher* dispatcher_; std::unique_ptr<Icmp> icmp_; const uint16_t echo_id_; // unique ID for this object's echo request/replies uint16_t current_sequence_number_; std::map<uint16_t, SentRecvTimePair> seq_num_to_sent_recv_time_; std::set<uint16_t> received_echo_reply_seq_numbers_; // Allow for an injectable tick clock for testing. base::TickClock* tick_clock_; base::DefaultTickClock default_tick_clock_; base::CancelableClosure timeout_callback_; IcmpSessionResultCallback result_callback_; IOHandler::InputCallback echo_reply_callback_; std::unique_ptr<IOHandler> echo_reply_handler_; DISALLOW_COPY_AND_ASSIGN(IcmpSession); }; } // namespace shill #endif // SHILL_ICMP_SESSION_H_