summaryrefslogtreecommitdiffstats
path: root/traceroute.cc
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--traceroute.cc1575
1 files changed, 1575 insertions, 0 deletions
diff --git a/traceroute.cc b/traceroute.cc
new file mode 100644
index 0000000..6a30296
--- /dev/null
+++ b/traceroute.cc
@@ -0,0 +1,1575 @@
+/***************************************************************************
+ * traceroute.cc -- Parallel multi-protocol traceroute feature *
+ * *
+ ***********************IMPORTANT NMAP LICENSE TERMS************************
+ *
+ * The Nmap Security Scanner is (C) 1996-2023 Nmap Software LLC ("The Nmap
+ * Project"). Nmap is also a registered trademark of the Nmap Project.
+ *
+ * This program is distributed under the terms of the Nmap Public Source
+ * License (NPSL). The exact license text applying to a particular Nmap
+ * release or source code control revision is contained in the LICENSE
+ * file distributed with that version of Nmap or source code control
+ * revision. More Nmap copyright/legal information is available from
+ * https://nmap.org/book/man-legal.html, and further information on the
+ * NPSL license itself can be found at https://nmap.org/npsl/ . This
+ * header summarizes some key points from the Nmap license, but is no
+ * substitute for the actual license text.
+ *
+ * Nmap is generally free for end users to download and use themselves,
+ * including commercial use. It is available from https://nmap.org.
+ *
+ * The Nmap license generally prohibits companies from using and
+ * redistributing Nmap in commercial products, but we sell a special Nmap
+ * OEM Edition with a more permissive license and special features for
+ * this purpose. See https://nmap.org/oem/
+ *
+ * If you have received a written Nmap license agreement or contract
+ * stating terms other than these (such as an Nmap OEM license), you may
+ * choose to use and redistribute Nmap under those terms instead.
+ *
+ * The official Nmap Windows builds include the Npcap software
+ * (https://npcap.com) for packet capture and transmission. It is under
+ * separate license terms which forbid redistribution without special
+ * permission. So the official Nmap Windows builds may not be redistributed
+ * without special permission (such as an Nmap OEM license).
+ *
+ * Source is provided to this software because we believe users have a
+ * right to know exactly what a program is going to do before they run it.
+ * This also allows you to audit the software for security holes.
+ *
+ * Source code also allows you to port Nmap to new platforms, fix bugs, and add
+ * new features. You are highly encouraged to submit your changes as a Github PR
+ * or by email to the dev@nmap.org mailing list for possible incorporation into
+ * the main distribution. Unless you specify otherwise, it is understood that
+ * you are offering us very broad rights to use your submissions as described in
+ * the Nmap Public Source License Contributor Agreement. This is important
+ * because we fund the project by selling licenses with various terms, and also
+ * because the inability to relicense code has caused devastating problems for
+ * other Free Software projects (such as KDE and NASM).
+ *
+ * The free version of Nmap is distributed in the hope that it will be
+ * useful, but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. Warranties,
+ * indemnification and commercial support are all available through the
+ * Npcap OEM program--see https://nmap.org/oem/
+ *
+ ***************************************************************************/
+
+/*
+Traceroute for Nmap. This traceroute is faster than a traditional traceroute
+because it sends several probes in parallel and detects shared traces.
+
+The algorithm works by sending probes with varying TTL values and waiting for
+TTL_EXCEEDED messages. As intermediate hops are discovered, they are entered
+into a global hop cache that is shared between targets and across host groups.
+When a hop is discovered and is found to be already in the cache, the trace for
+that target is linked into the cached trace and there is no need to try lower
+TTLs. The process results in the building of a tree of Hop structures.
+
+The order in which probes are sent does not matter to the accuracy of the
+algorithm but it does matter to the speed. The sooner a shared trace can be
+detected, and the higher the TTL at which it is detected, the fewer probes need
+to be sent. The ideal situation is to start sending probes with a TTL equal to
+the true distance and count downward from there. In that case it may only be
+necessary to send two probes per target: one at the distance of the target to
+get a response, and one at distance - 1 to get a cache hit. When the distance
+isn't known in advance, the algorithm arbitrarily starts at a TTL of 10 and
+counts downward, then counts upward from 11 until it reaches the target. So a
+typical trace may look like
+
+TTL 10 -> TTL_EXCEEDED
+TTL 9 -> TTL_EXCEEDED
+TTL 8 -> TTL_EXCEEDED
+TTL 7 -> cache hit
+TTL 11 -> TTL_EXCEEDED
+TTL 12 -> TTL_EXCEEDED
+TTL 13 -> SYN/ACK, or whatever is the target's response to the probe
+
+The output for this host would then say "Hops 1-7 are the same as for ...".
+
+The detection of shared traces rests on the assumption that all paths going
+through a router at a certain TTL will be identical up to and including the
+router. This assumption is not always true. Even if two targets are each one hop
+past router X at TTL 10, packets may follow different paths to each host (and
+those paths may even change over time). This traceroute algorithm will be fooled
+by such a situation, and will report that the paths are identical up to
+router X. The only way to be sure is to do a complete trace for each target
+individually.
+*/
+
+#include "nmap_dns.h"
+#include "nmap_error.h"
+#include "nmap_tty.h"
+#include "osscan2.h"
+#include "payload.h"
+#include "timing.h"
+#include "NmapOps.h"
+#include "Target.h"
+#include "tcpip.h"
+
+#include "struct_ip.h"
+
+#ifndef IPPROTO_SCTP
+#include "libnetutil/netutil.h"
+#endif
+
+#include <dnet.h>
+
+#include <algorithm>
+#include <list>
+#include <map>
+#include <set>
+#include <vector>
+
+extern NmapOps o;
+
+/* The highest TTL we go up to if the target itself doesn't respond. */
+#define MAX_TTL 30
+#define MAX_OUTSTANDING_PROBES 10
+#define MAX_RESENDS 2
+/* In milliseconds. */
+#define PROBE_TIMEOUT 1000
+/* If the hop cache (including timed-out hops) is bigger than this after a
+ round, the hop is cleared and rebuilt from scratch. */
+#define MAX_HOP_CACHE_SIZE 1000
+
+struct Hop;
+class HostState;
+class Probe;
+
+/* An object of this class is a (TTL, address) pair that uniquely identifies a
+ hop. Hops in the hop_cache are indexed by this type. */
+struct HopIdent {
+ u8 ttl;
+ struct sockaddr_storage addr;
+
+ HopIdent(u8 ttl, const struct sockaddr_storage &addr) {
+ this->addr = addr;
+ this->ttl = ttl;
+ }
+
+ bool operator<(const struct HopIdent &other) const {
+ if (ttl < other.ttl)
+ return true;
+ else if (ttl > other.ttl)
+ return false;
+ else
+ return sockaddr_storage_cmp(&addr, &other.addr) < 0;
+ }
+};
+
+/* A global random token used to distinguish this traceroute's probes from
+ those of other traceroutes possibly running on the same machine. */
+static u16 global_id;
+/* A global cache of known hops, indexed by TTL and address. */
+static std::map<struct HopIdent, Hop *> hop_cache;
+/* A list of timedout hops, which are not kept in hop_cache, so we can delete
+ all hops on occasion. */
+/* This would be stack-allocated except for a weird bug on AIX that causes
+ * infinite loops when trying to traverse the list. For some reason,
+ * dynamically allocating it fixes the bug. */
+static std::list<Hop *> *timedout_hops = NULL;
+/* The TTL at which we start sending probes if we don't have a distance
+ estimate. This is updated after each host group on the assumption that hosts
+ across groups will not differ much in distance. Having this closer to the
+ true distance makes the trace faster but is not needed for accuracy. */
+static u8 initial_ttl = 10;
+
+static struct timeval get_now(struct timeval *now = NULL);
+static const char *ss_to_string(const struct sockaddr_storage *ss);
+
+struct Hop {
+ Hop *parent;
+ struct sockaddr_storage tag;
+ /* When addr.ss_family == 0, this hop represents a timeout. */
+ struct sockaddr_storage addr;
+ u8 ttl;
+ float rtt; /* In milliseconds. */
+ std::string hostname;
+
+ Hop() {
+ this->parent = NULL;
+ this->addr.ss_family = 0;
+ this->ttl = 0;
+ this->rtt = -1.0;
+ this->tag.ss_family = 0;
+ }
+
+ Hop(u8 ttl, const struct sockaddr_storage &addr, float rtt) {
+ this->parent = NULL;
+ this->addr = addr;
+ this->ttl = ttl;
+ this->rtt = rtt;
+ this->tag.ss_family = 0;
+ }
+};
+
+class HostState {
+public:
+ enum counting_state { COUNTING_DOWN, COUNTING_UP };
+
+ Target *target;
+ /* A bitmap of TTLs that have been sent, to avoid duplicates when we switch
+ around the order counting up or down. */
+ std::vector<bool> sent_ttls;
+ u8 current_ttl;
+ enum counting_state state;
+ /* If nonzero, the known hop distance to the target. */
+ int reached_target;
+ struct probespec pspec;
+ std::list<Probe *> unanswered_probes;
+ std::list<Probe *> active_probes;
+ std::list<Probe *> pending_resends;
+ Hop *hops;
+
+ HostState(Target *target);
+ ~HostState();
+ bool has_more_probes() const;
+ bool is_finished() const;
+ bool send_next_probe(int rawsd, eth_t *ethsd);
+ void next_ttl();
+ void count_up();
+ int cancel_probe(std::list<Probe *>::iterator it);
+ int cancel_probes_below(u8 ttl);
+ int cancel_probes_above(u8 ttl);
+ Hop *insert_hop(u8 ttl, const struct sockaddr_storage *addr, float rtt);
+ void link_to(Hop *hop);
+ double completion_fraction() const;
+
+private:
+ void child_parent_ttl(u8 ttl, Hop **child, Hop **parent) const;
+ static u8 distance_guess(const Target *target);
+ static struct probespec get_probe(const Target *target);
+};
+
+class Probe {
+private:
+ /* This is incremented with each instantiated probe. */
+ static u16 token_counter;
+
+ unsigned int num_resends;
+
+public:
+ HostState *host;
+ struct probespec pspec;
+ u8 ttl;
+ /* The token is used to match up probe replies. */
+ u16 token;
+ struct timeval sent_time;
+
+ Probe(HostState *host, struct probespec pspec, u8 ttl);
+ virtual ~Probe();
+ void send(int rawsd, eth_t *ethsd, struct timeval *now = NULL);
+ void resend(int rawsd, eth_t *ethsd, struct timeval *now = NULL);
+ bool is_timedout(struct timeval *now = NULL) const;
+ bool may_resend() const;
+ virtual unsigned char *build_packet(const struct sockaddr_storage *source,
+ u32 *len) const = 0;
+
+ static Probe *make(HostState *host, struct probespec pspec, u8 ttl);
+};
+u16 Probe::token_counter = 0x0000;
+
+class TracerouteState {
+public:
+ std::list<HostState *> active_hosts;
+ /* The next send time for enforcing scan delay. */
+ struct timeval next_send_time;
+
+ TracerouteState(std::vector<Target *> &targets);
+ ~TracerouteState();
+
+ void send_new_probes();
+ void read_replies(long timeout);
+ void cull_timeouts();
+ void remove_finished_hosts();
+ void resolve_hops();
+ void transfer_hops();
+
+ double completion_fraction() const;
+
+private:
+ eth_t *ethsd;
+ int rawsd;
+ pcap_t *pd;
+ int num_active_probes;
+
+ std::vector<HostState *> hosts;
+ std::list<HostState *>::iterator next_sending_host;
+
+ void next_active_host();
+ Probe *lookup_probe(const struct sockaddr_storage *target_addr, u16 token);
+ void set_host_hop(HostState *host, u8 ttl,
+ const struct sockaddr_storage *from_addr, float rtt);
+ void set_host_hop_timedout(HostState *host, u8 ttl);
+};
+
+static Hop *merge_hops(const struct sockaddr_storage *tag, Hop *a, Hop *b);
+static Hop *hop_cache_lookup(u8 ttl, const struct sockaddr_storage *addr);
+static void hop_cache_insert(Hop *hop);
+static unsigned int hop_cache_size();
+
+HostState::HostState(Target *target) : sent_ttls(MAX_TTL + 1, false) {
+ this->target = target;
+ current_ttl = MIN(MAX(1, HostState::distance_guess(target)), MAX_TTL);
+ state = HostState::COUNTING_DOWN;
+ reached_target = 0;
+ pspec = HostState::get_probe(target);
+ hops = NULL;
+}
+
+HostState::~HostState() {
+ /* active_probes and pending_resends are subsets of unanswered_probes, so we
+ delete the allocated probes in unanswered_probes only. */
+ while (!unanswered_probes.empty()) {
+ delete *unanswered_probes.begin();
+ unanswered_probes.pop_front();
+ }
+ while (!active_probes.empty())
+ active_probes.pop_front();
+ while (!pending_resends.empty())
+ pending_resends.pop_front();
+}
+
+bool HostState::has_more_probes() const {
+ /* We are done if we are counting up and
+ 1. we've reached and exceeded the target, or
+ 2. we've exceeded MAX_TTL. */
+ return !(state == HostState::COUNTING_UP
+ && ((reached_target > 0 && current_ttl >= reached_target)
+ || current_ttl > MAX_TTL));
+}
+
+bool HostState::is_finished() const {
+ return !this->has_more_probes()
+ && active_probes.empty() && pending_resends.empty();
+}
+
+bool HostState::send_next_probe(int rawsd, eth_t *ethsd) {
+ Probe *probe;
+
+ /* Do a resend if possible. */
+ if (!pending_resends.empty()) {
+ probe = pending_resends.front();
+ pending_resends.pop_front();
+ active_probes.push_back(probe);
+ probe->resend(rawsd, ethsd);
+ return true;
+ }
+
+ this->next_ttl();
+
+ if (!this->has_more_probes())
+ return false;
+
+ probe = Probe::make(this, pspec, current_ttl);
+ unanswered_probes.push_back(probe);
+ active_probes.push_back(probe);
+ probe->send(rawsd, ethsd);
+ sent_ttls[current_ttl] = true;
+
+ return true;
+}
+
+/* Find the next TTL we should send to. */
+void HostState::next_ttl() {
+ assert(current_ttl > 0);
+ if (state == HostState::COUNTING_DOWN) {
+ while (current_ttl > 1 && sent_ttls[current_ttl])
+ current_ttl--;
+ if (current_ttl == 1)
+ state = HostState::COUNTING_UP;
+ }
+ /* Note no "else". */
+ if (state == HostState::COUNTING_UP) {
+ while (current_ttl <= MAX_TTL && sent_ttls[current_ttl])
+ current_ttl++;
+ }
+}
+
+int HostState::cancel_probe(std::list<Probe *>::iterator it) {
+ int count;
+
+ count = active_probes.size();
+ active_probes.remove(*it);
+ count -= active_probes.size();
+ pending_resends.remove(*it);
+ delete *it;
+ unanswered_probes.erase(it);
+
+ return count;
+}
+
+int HostState::cancel_probes_below(u8 ttl) {
+ std::list<Probe *>::iterator it, next;
+ int count;
+
+ count = 0;
+ for (it = unanswered_probes.begin(); it != unanswered_probes.end(); it = next) {
+ next = it;
+ next++;
+ if ((*it)->ttl < ttl)
+ count += this->cancel_probe(it);
+ }
+
+ return count;
+}
+
+int HostState::cancel_probes_above(u8 ttl) {
+ std::list<Probe *>::iterator it, next;
+ int count;
+
+ count = 0;
+ for (it = unanswered_probes.begin(); it != unanswered_probes.end(); it = next) {
+ next = it;
+ next++;
+ if ((*it)->ttl > ttl)
+ count += this->cancel_probe(it);
+ }
+
+ return count;
+}
+
+Hop *HostState::insert_hop(u8 ttl, const struct sockaddr_storage *addr,
+ float rtt) {
+ Hop *hop, *prev, *p;
+
+ this->child_parent_ttl(ttl, &prev, &p);
+ if (p != NULL && p->ttl == ttl) {
+ hop = p;
+ /* Collision with the same TTL and a different address. */
+ if (hop->addr.ss_family == 0) {
+ /* Hit a timed-out hop. Fill in the missing address and RTT. */
+ hop->addr = *addr;
+ hop->rtt = rtt;
+ } else {
+ if (o.debugging) {
+ log_write(LOG_STDOUT, "Found existing %s", ss_to_string(&hop->addr));
+ log_write(LOG_STDOUT, " while inserting %s at TTL %d for %s\n",
+ ss_to_string(addr), ttl, target->targetipstr());
+ }
+ }
+ } else {
+ hop = new Hop(ttl, *addr, rtt);
+ hop->parent = p;
+ if (prev == NULL) {
+ size_t sslen;
+ this->hops = hop;
+ sslen = sizeof(hop->tag);
+ target->TargetSockAddr(&hop->tag, &sslen);
+ } else {
+ prev->parent = hop;
+ hop->tag = prev->tag;
+ }
+ hop_cache_insert(hop);
+ }
+
+ return hop;
+}
+
+void HostState::link_to(Hop *hop) {
+ Hop *prev, *p;
+
+ this->child_parent_ttl(hop->ttl, &prev, &p);
+ if (hop == p) {
+ /* Already linked for this host. This can happen a reply for a higher TTL
+ results in a merge, and later a reply for a lower TTL comes back. */
+ return;
+ }
+ if (o.debugging > 1) {
+ log_write(LOG_STDOUT, "Merging traces below TTL %d for %s",
+ hop->ttl, ss_to_string(&hop->tag));
+ log_write(LOG_STDOUT, " and %s\n", target->targetipstr());
+ }
+ hop = merge_hops(&hop->tag, hop, p);
+ if (prev == NULL)
+ this->hops = hop;
+ else
+ prev->parent = hop;
+}
+
+double HostState::completion_fraction() const {
+ unsigned int i, n;
+
+ if (this->is_finished())
+ return 1.0;
+
+ n = 0;
+ for (i = 0; i < sent_ttls.size(); i++) {
+ if (sent_ttls[i])
+ n++;
+ }
+
+ return (double) n / sent_ttls.size();
+}
+
+void HostState::child_parent_ttl(u8 ttl, Hop **child, Hop **parent) const {
+ *child = NULL;
+ *parent = this->hops;
+ while (*parent != NULL && (*parent)->ttl > ttl) {
+ *child = *parent;
+ *parent = (*parent)->parent;
+ }
+}
+
+u8 HostState::distance_guess(const Target *target) {
+ /* Use the distance from OS detection if we have it. */
+ if (target->distance != -1)
+ return target->distance;
+ else
+ /* initial_ttl is a variable with file-level scope. */
+ return initial_ttl;
+}
+
+/* Get the probe that will be used for the traceroute. This is the
+ highest-quality probe found in ping or port scanning, or ICMP echo if no
+ responsive probe is known. */
+struct probespec HostState::get_probe(const Target *target) {
+ struct probespec probe;
+
+ probe = target->pingprobe;
+ if (target->af() == AF_INET &&
+ (probe.type == PS_TCP || probe.type == PS_UDP || probe.type == PS_SCTP || probe.type == PS_ICMP)) {
+ /* Nothing needed. */
+ } else if (target->af() == AF_INET6 &&
+ (probe.type == PS_TCP || probe.type == PS_UDP || probe.type == PS_SCTP || probe.type == PS_ICMPV6)) {
+ /* Nothing needed. */
+ } else if (probe.type == PS_PROTO) {
+ /* If this is an IP protocol probe, fill in some fields for some common
+ protocols. We cheat and store them in the TCP-, UDP-, SCTP- and
+ ICMP-specific fields. */
+ if (probe.proto == IPPROTO_TCP) {
+ probe.pd.tcp.flags = TH_ACK;
+ probe.pd.tcp.dport = get_random_u16();
+ } else if (probe.proto == IPPROTO_UDP) {
+ probe.pd.udp.dport = get_random_u16();
+ } else if (probe.proto == IPPROTO_SCTP) {
+ probe.pd.sctp.dport = get_random_u16();
+ } else if (probe.proto == IPPROTO_ICMP) {
+ probe.pd.icmp.type = ICMP_ECHO;
+ } else if (probe.proto == IPPROTO_ICMPV6) {
+ probe.pd.icmp.type = ICMPV6_ECHO;
+ } else {
+ fatal("Unknown protocol %d", probe.proto);
+ }
+ } else {
+ /* No responsive probe known? The user probably skipped both ping and
+ port scan. Guess ICMP echo as the most likely to get a response. */
+ if (target->af() == AF_INET) {
+ probe.type = PS_ICMP;
+ probe.proto = IPPROTO_ICMP;
+ probe.pd.icmp.type = ICMP_ECHO;
+ probe.pd.icmp.code = 0;
+ } else if (target->af() == AF_INET6) {
+ probe.type = PS_ICMPV6;
+ probe.proto = IPPROTO_ICMPV6;
+ probe.pd.icmp.type = ICMPV6_ECHO;
+ probe.pd.icmp.code = 0;
+ } else {
+ fatal("Unknown address family %d", target->af());
+ }
+ }
+
+ return probe;
+}
+
+Probe::Probe(HostState *host, struct probespec pspec, u8 ttl) {
+ this->host = host;
+ this->pspec = pspec;
+ this->ttl = ttl;
+ token = Probe::token_counter++;
+ sent_time.tv_sec = 0;
+ sent_time.tv_usec = 0;
+ num_resends = 0;
+}
+
+Probe::~Probe() {
+}
+
+void Probe::send(int rawsd, eth_t *ethsd, struct timeval *now) {
+ struct eth_nfo eth;
+ struct eth_nfo *ethp;
+ int decoy;
+
+ /* Set up the Ethernet handle if we're using that. */
+ if (ethsd != NULL) {
+ memcpy(eth.srcmac, host->target->SrcMACAddress(), 6);
+ memcpy(eth.dstmac, host->target->NextHopMACAddress(), 6);
+ eth.ethsd = ethsd;
+ eth.devname[0] = '\0';
+ ethp = &eth;
+ } else {
+ ethp = NULL;
+ }
+
+ for (decoy = 0; decoy < o.numdecoys; decoy++) {
+ const struct sockaddr_storage *source;
+ unsigned char *packet;
+ u32 packetlen;
+
+ if (decoy == o.decoyturn) {
+ source = host->target->SourceSockAddr();
+ sent_time = get_now(now);
+ } else {
+ source = &(o.decoys[decoy]);
+ }
+
+ packet = this->build_packet(source, &packetlen);
+ send_ip_packet(rawsd, ethp, host->target->TargetSockAddr(), packet, packetlen);
+ free(packet);
+ }
+}
+
+void Probe::resend(int rawsd, eth_t *ethsd, struct timeval *now) {
+ num_resends++;
+ this->send(rawsd, ethsd, now);
+}
+
+bool Probe::is_timedout(struct timeval *now) const {
+ struct timeval tv;
+
+ tv = get_now(now);
+
+ return TIMEVAL_MSEC_SUBTRACT(tv, sent_time) > PROBE_TIMEOUT;
+}
+
+bool Probe::may_resend() const {
+ return num_resends < MIN(o.getMaxRetransmissions(), MAX_RESENDS);
+}
+
+/* Probe is an abstract class with a missing build_packet method. These concrete
+ subclasses implement the method for different probe types. */
+
+class ICMPProbe : public Probe
+{
+public:
+ ICMPProbe(HostState *host, struct probespec pspec, u8 ttl)
+ : Probe(host, pspec, ttl) {
+ }
+
+ unsigned char *build_packet(const struct sockaddr_storage *source, u32 *len) const {
+ const struct sockaddr_in *sin;
+ assert(source->ss_family == AF_INET);
+ sin = (struct sockaddr_in *) source;
+ return build_icmp_raw(&sin->sin_addr, host->target->v4hostip(), ttl,
+ 0x0000, 0x00, false, NULL, 0, token, global_id,
+ pspec.pd.icmp.type, pspec.pd.icmp.code,
+ o.extra_payload, o.extra_payload_length, len);
+ }
+};
+
+class TCPProbe : public Probe
+{
+public:
+ TCPProbe(HostState *host, struct probespec pspec, u8 ttl)
+ : Probe(host, pspec, ttl) {
+ }
+ unsigned char *build_packet(const struct sockaddr_storage *source, u32 *len) const {
+ const char *tcpopts;
+ int tcpoptslen;
+ u32 ack;
+
+ tcpopts = NULL;
+ tcpoptslen = 0;
+ ack = 0;
+ if ((pspec.pd.tcp.flags & TH_SYN) == TH_SYN) {
+ /* MSS 1460 bytes. */
+ tcpopts = TCP_SYN_PROBE_OPTIONS;
+ tcpoptslen = TCP_SYN_PROBE_OPTIONS_LEN;
+ } else if ((pspec.pd.tcp.flags & TH_ACK) == TH_ACK) {
+ ack = get_random_u32();
+ }
+
+ /* For TCP we encode the token in the source port. */
+ if (source->ss_family == AF_INET) {
+ const struct sockaddr_in *sin = (struct sockaddr_in *) source;
+ return build_tcp_raw(&sin->sin_addr, host->target->v4hostip(), ttl,
+ get_random_u16(), get_random_u8(), false, NULL, 0,
+ token ^ global_id, pspec.pd.tcp.dport, get_random_u32(), ack, 0x00,
+ pspec.pd.tcp.flags, get_random_u16(), 0, (const u8 *) tcpopts, tcpoptslen,
+ o.extra_payload, o.extra_payload_length, len);
+ } else if (source->ss_family == AF_INET6) {
+ const struct sockaddr_in6 *sin6 = (struct sockaddr_in6 *) source;
+ return build_tcp_raw_ipv6(&sin6->sin6_addr, host->target->v6hostip(),
+ 0, 0, ttl,
+ token ^ global_id, pspec.pd.tcp.dport, get_random_u32(), ack, 0x00,
+ pspec.pd.tcp.flags, get_random_u16(), 0, (const u8 *) tcpopts, tcpoptslen,
+ o.extra_payload, o.extra_payload_length, len);
+ } else {
+ fatal("Unknown address family %u in %s.", source->ss_family, __func__);
+ }
+ }
+};
+
+class UDPProbe : public Probe
+{
+public:
+ UDPProbe(HostState *host, struct probespec pspec, u8 ttl)
+ : Probe(host, pspec, ttl) {
+ }
+ unsigned char *build_packet(const struct sockaddr_storage *source, u32 *len) const {
+ const u8 *payload;
+ int payload_length;
+
+ payload = get_udp_payload(pspec.pd.udp.dport, &payload_length, 0);
+
+ /* For UDP we encode the token in the source port. */
+ if (source->ss_family == AF_INET) {
+ const struct sockaddr_in *sin = (struct sockaddr_in *) source;
+ return build_udp_raw(&sin->sin_addr, host->target->v4hostip(), ttl,
+ get_random_u16(), get_random_u8(), false, NULL, 0,
+ token ^ global_id, pspec.pd.udp.dport,
+ (char *) payload, payload_length, len);
+ } else if (source->ss_family == AF_INET6) {
+ const struct sockaddr_in6 *sin6 = (struct sockaddr_in6 *) source;
+ return build_udp_raw_ipv6(&sin6->sin6_addr, host->target->v6hostip(),
+ 0, 0, ttl,
+ token ^ global_id, pspec.pd.udp.dport,
+ (char *) payload, payload_length, len);
+ } else {
+ fatal("Unknown address family %u in %s.", source->ss_family, __func__);
+ }
+ }
+};
+
+class SCTPProbe : public Probe
+{
+public:
+ SCTPProbe(HostState *host, struct probespec pspec, u8 ttl)
+ : Probe(host, pspec, ttl) {
+ }
+ unsigned char *build_packet(const struct sockaddr_storage *source, u32 *len) const {
+ struct sctp_chunkhdr_init chunk;
+
+ sctp_pack_chunkhdr_init(&chunk, SCTP_INIT, 0, sizeof(chunk),
+ get_random_u32() /*itag*/, 32768, 10, 2048, get_random_u32() /*itsn*/);
+
+ if (source->ss_family == AF_INET) {
+ const struct sockaddr_in *sin = (struct sockaddr_in *) source;
+ return build_sctp_raw(&sin->sin_addr, host->target->v4hostip(), ttl,
+ get_random_u16(), get_random_u8(), false, NULL, 0,
+ token ^ global_id, pspec.pd.sctp.dport, 0UL,
+ (char *) &chunk, sizeof(chunk),
+ o.extra_payload, o.extra_payload_length, len);
+ } else if (source->ss_family == AF_INET6) {
+ const struct sockaddr_in6 *sin6 = (struct sockaddr_in6 *) source;
+ return build_sctp_raw_ipv6(&sin6->sin6_addr, host->target->v6hostip(),
+ 0, 0, ttl,
+ token ^ global_id, pspec.pd.sctp.dport, 0UL,
+ (char *) &chunk, sizeof(chunk),
+ o.extra_payload, o.extra_payload_length, len);
+ } else {
+ fatal("Unknown address family %u in %s.", source->ss_family, __func__);
+ }
+ }
+};
+
+class IPProtoProbe : public Probe
+{
+public:
+ IPProtoProbe(HostState *host, struct probespec pspec, u8 ttl)
+ : Probe(host, pspec, ttl) {
+ }
+ unsigned char *build_packet(const struct sockaddr_storage *source, u32 *len) const {
+ /* For IP proto scan the token is put in the IP ID or flow label. */
+ if (source->ss_family == AF_INET) {
+ const struct sockaddr_in *sin = (struct sockaddr_in *) source;
+ return build_ip_raw(&sin->sin_addr, host->target->v4hostip(), pspec.proto, ttl,
+ token ^ global_id, get_random_u8(), false, NULL, 0,
+ o.extra_payload, o.extra_payload_length, len);
+ } else if (source->ss_family == AF_INET6) {
+ const struct sockaddr_in6 *sin6 = (struct sockaddr_in6 *) source;
+ return build_ipv6_raw(&sin6->sin6_addr, host->target->v6hostip(),
+ 0, token ^ global_id, pspec.proto, ttl,
+ o.extra_payload, o.extra_payload_length, len);
+ } else {
+ fatal("Unknown address family %u in %s.", source->ss_family, __func__);
+ }
+ }
+};
+
+class ICMPv6Probe : public Probe
+{
+public:
+ ICMPv6Probe(HostState *host, struct probespec pspec, u8 ttl)
+ : Probe(host, pspec, ttl) {
+ }
+
+ unsigned char *build_packet(const struct sockaddr_storage *source, u32 *len) const {
+ const struct sockaddr_in6 *sin6;
+ assert(source->ss_family == AF_INET6);
+ sin6 = (struct sockaddr_in6 *) source;
+ return build_icmpv6_raw(&sin6->sin6_addr, host->target->v6hostip(), 0x00, 0x0000,
+ ttl, token, global_id, pspec.pd.icmp.type, pspec.pd.icmp.code,
+ o.extra_payload, o.extra_payload_length, len);
+ }
+};
+
+Probe *Probe::make(HostState *host, struct probespec pspec, u8 ttl)
+{
+ if (pspec.type == PS_ICMP || (pspec.type == PS_PROTO && pspec.proto == IPPROTO_ICMP))
+ return new ICMPProbe(host, pspec, ttl);
+ else if (pspec.type == PS_TCP || (pspec.type == PS_PROTO && pspec.proto == IPPROTO_TCP))
+ return new TCPProbe(host, pspec, ttl);
+ else if (pspec.type == PS_UDP || (pspec.type == PS_PROTO && pspec.proto == IPPROTO_UDP))
+ return new UDPProbe(host, pspec, ttl);
+ else if (pspec.type == PS_SCTP || (pspec.type == PS_PROTO && pspec.proto == IPPROTO_SCTP))
+ return new SCTPProbe(host, pspec, ttl);
+ else if (pspec.type == PS_PROTO)
+ return new IPProtoProbe(host, pspec, ttl);
+ else if (pspec.type == PS_ICMPV6)
+ return new ICMPv6Probe(host, pspec, ttl);
+ else
+ fatal("Unknown probespec type in traceroute");
+
+ return NULL;
+}
+
+TracerouteState::TracerouteState(std::vector<Target *> &targets) {
+ std::vector<Target *>::const_iterator it;
+ char pcap_filter[128];
+ int n;
+
+ assert(targets.size() > 0);
+
+ if ((o.sendpref & PACKET_SEND_ETH) && targets[0]->ifType() == devt_ethernet) {
+ /* No need to check for g_has_npcap_loopback on WIN32 because devt_loopback
+ * is checked earlier. */
+ ethsd = eth_open_cached(targets[0]->deviceName());
+ if (ethsd == NULL)
+ fatal("dnet: failed to open device %s", targets[0]->deviceName());
+ rawsd = -1;
+ } else {
+#ifdef WIN32
+ win32_fatal_raw_sockets(targets[0]->deviceName());
+#endif
+ rawsd = nmap_raw_socket();
+ if (rawsd < 0)
+ pfatal("traceroute: socket troubles");
+ ethsd = NULL;
+ }
+
+ /* Assume that all the targets share the same device. */
+ if((pd=my_pcap_open_live(targets[0]->deviceName(), 128, o.spoofsource, 2))==NULL)
+ fatal("%s", PCAP_OPEN_ERRMSG);
+ n = Snprintf(pcap_filter, sizeof(pcap_filter), "(ip or ip6) and dst host %s",
+ ss_to_string(targets[0]->SourceSockAddr()));
+ assert(n < (int) sizeof(pcap_filter));
+ set_pcap_filter(targets[0]->deviceFullName(), pd, pcap_filter);
+ if (o.debugging)
+ log_write(LOG_STDOUT, "Packet capture filter (device %s): %s\n", targets[0]->deviceFullName(), pcap_filter);
+ for (it = targets.begin(); it != targets.end(); it++) {
+ HostState *state = new HostState(*it);
+ hosts.push_back(state);
+ active_hosts.push_back(state);
+ }
+
+ num_active_probes = 0;
+ next_sending_host = active_hosts.begin();
+ next_send_time = get_now();
+}
+
+TracerouteState::~TracerouteState() {
+ std::vector<HostState *>::iterator it;
+
+ if (rawsd != -1)
+ close(rawsd);
+ pcap_close(pd);
+
+ for (it = hosts.begin(); it != hosts.end(); it++)
+ delete *it;
+}
+
+void TracerouteState::next_active_host() {
+ assert(next_sending_host != active_hosts.end());
+ next_sending_host++;
+ if (next_sending_host == active_hosts.end())
+ next_sending_host = active_hosts.begin();
+}
+
+void TracerouteState::send_new_probes() {
+ std::list<HostState *>::const_iterator failed_host;
+ struct timeval now;
+
+ now = get_now();
+
+ assert(!active_hosts.empty());
+ failed_host = active_hosts.end();
+ while (next_sending_host != failed_host
+ && num_active_probes < MAX_OUTSTANDING_PROBES
+ && !TIMEVAL_BEFORE(now, next_send_time)) {
+ if ((*next_sending_host)->send_next_probe(rawsd, ethsd)) {
+ num_active_probes++;
+ TIMEVAL_MSEC_ADD(next_send_time, next_send_time, o.scan_delay);
+ if (TIMEVAL_BEFORE(next_send_time, now))
+ next_send_time = now;
+ failed_host = active_hosts.end();
+ } else if (failed_host == active_hosts.end()) {
+ failed_host = next_sending_host;
+ }
+ next_active_host();
+ }
+}
+
+static Hop *hop_cache_lookup(u8 ttl, const struct sockaddr_storage *addr) {
+ std::map<struct HopIdent, Hop *>::iterator it;
+ HopIdent ident(ttl, *addr);
+
+ it = hop_cache.find(ident);
+ if (it == hop_cache.end())
+ return NULL;
+ else
+ return it->second;
+}
+
+static void hop_cache_insert(Hop *hop) {
+ if (hop->addr.ss_family == 0) {
+ timedout_hops->push_back(hop);
+ } else {
+ hop_cache[HopIdent(hop->ttl, hop->addr)] = hop;
+ }
+}
+
+static unsigned int hop_cache_size() {
+ return hop_cache.size() + timedout_hops->size();
+}
+
+void traceroute_hop_cache_clear() {
+ std::map<struct HopIdent, Hop *>::iterator map_iter;
+ std::list<Hop *>::iterator list_iter;
+
+ for (map_iter = hop_cache.begin(); map_iter != hop_cache.end(); map_iter++)
+ delete map_iter->second;
+ hop_cache.clear();
+ if (!timedout_hops) return;
+ for (list_iter = timedout_hops->begin(); list_iter != timedout_hops->end(); list_iter++)
+ delete *list_iter;
+ timedout_hops->clear();
+}
+
+/* Merge two hop chains together and return the head of the merged chain. This
+ is done when a cache hit finds that two targets share the same intermediate
+ hop; rather than doing a full trace for each target, one is linked to the
+ other. "Merged" means that, for example, if chain a has an hop at TTL 3 and
+ chain b has one for TTL 2, the merged chain will include both. Usually, the
+ chains will not have hops at the same TTL (that implies different routes to
+ the same host), but see the next paragraph for what we do when that happens.
+ Each hop in the merged chain will be tagged with the given tag.
+
+ There are many cases that must be handled correctly by this function: a and b
+ may be equal; either may be NULL; a and b may be disjoint chains or may be
+ joined somewhere. The biggest difficulty is when both of the chains have a
+ hop with the same TTL but a different address. When this happens we
+ arbitrarily choose one of the hops to unlink, on the presumption that any
+ route through the same intermediate host at a given TTL should be the same
+ and that differences aren't meaningful. (This has the same effect as if we
+ were to send probes strictly serially, because then there would be no parent
+ hops to potentially conflict, even if in fact they would if traced to
+ completion.) */
+static Hop *merge_hops(const struct sockaddr_storage *tag, Hop *a, Hop *b) {
+ Hop head, *p;
+
+ p = &head;
+
+ while (a != NULL && b != NULL && a != b) {
+ Hop **next;
+
+ if (a->ttl > b->ttl) {
+ next = &a;
+ } else if (b->ttl > a->ttl) {
+ next = &b;
+ } else {
+ Hop **discard, *parent;
+
+ if (b->addr.ss_family == 0) {
+ next = &a;
+ discard = &b;
+ } else if (a->addr.ss_family == 0) {
+ next = &b;
+ discard = &a;
+ } else {
+ next = &a;
+ discard = &b;
+ if (o.debugging) {
+ log_write(LOG_STDOUT, "Warning: %s", ss_to_string(&(*next)->addr));
+ log_write(LOG_STDOUT, " doesn't match %s at TTL %d;",
+ ss_to_string(&(*discard)->addr), a->ttl);
+ log_write(LOG_STDOUT, " arbitrarily discarding %s\n",
+ ss_to_string(&(*discard)->addr));
+ }
+ }
+ parent = (*discard)->parent;
+ *discard = parent;
+ }
+ p->parent = *next;
+ p->tag = *tag;
+ p = p->parent;
+ *next = (*next)->parent;
+ }
+ /* At most one branch of this is taken, even when a == b. */
+ if (a != NULL)
+ p->parent = a;
+ else if (b != NULL)
+ p->parent = b;
+ for (; p != NULL; p = p->parent)
+ p->tag = *tag;
+
+ return head.parent;
+}
+
+/* Record a hop at the given TTL for the given host. This takes care of linking
+ the hop into the host's chain as well as into the global hop tree. */
+void TracerouteState::set_host_hop(HostState *host, u8 ttl,
+ const struct sockaddr_storage *from_addr, float rtt) {
+ Hop *hop;
+
+ if (o.debugging > 1) {
+ log_write(LOG_STDOUT, "Set hop %s TTL %d to %s RTT %.2f ms\n",
+ host->target->targetipstr(), ttl, ss_to_string(from_addr), rtt);
+ }
+
+ hop = hop_cache_lookup(ttl, from_addr);
+ if (hop == NULL) {
+ /* A new hop, never before seen with this address and TTL. Add it to the
+ host's chain and to the global cache. */
+ hop = host->insert_hop(ttl, from_addr, rtt);
+ } else {
+ /* An existing hop at this address and TTL. Link this host's chain to it. */
+ if (o.debugging > 1) {
+ log_write(LOG_STDOUT, "Traceroute cache hit %s TTL %d while tracing",
+ ss_to_string(&hop->addr), hop->ttl);
+ log_write(LOG_STDOUT, " %s TTL %d\n", host->target->targetipstr(), ttl);
+ }
+
+ host->link_to(hop);
+
+ if (host->state == HostState::COUNTING_DOWN) {
+ /* Hit the cache going down. Seek to the end of the chain. If we have the
+ tag for the last node, we take responsibility for finishing the trace.
+ Otherwise, start counting up. */
+
+ while (hop->parent != NULL) {
+ hop = hop->parent;
+ /* No need to re-probe any merged hops. */
+ host->sent_ttls[hop->ttl] = true;
+ }
+ if (sockaddr_storage_equal(&hop->tag, host->target->TargetSockAddr())) {
+ if (o.debugging > 1) {
+ log_write(LOG_STDOUT, "%s continuing trace from TTL %d\n",
+ host->target->targetipstr(), host->current_ttl);
+ }
+ } else {
+ host->state = HostState::COUNTING_UP;
+ num_active_probes -= host->cancel_probes_below(ttl);
+ }
+ }
+ }
+}
+
+/* Record that a hop at the given TTL for the given host timed out. */
+void TracerouteState::set_host_hop_timedout(HostState *host, u8 ttl) {
+ static struct sockaddr_storage EMPTY_ADDR = { 0 };
+ host->insert_hop(ttl, &EMPTY_ADDR, -1.0);
+}
+
+struct Reply {
+ struct timeval rcvdtime;
+ struct sockaddr_storage from_addr;
+ struct sockaddr_storage target_addr;
+ u8 ttl;
+ u16 token;
+};
+
+static bool parse_encapsulated_reply(const void *ip, unsigned len, Reply *reply) {
+ struct abstract_ip_hdr hdr;
+ const void *data;
+
+ data = ip_get_data(ip, &len, &hdr);
+ if (data == NULL)
+ return false;
+
+ if (hdr.version == 4 && hdr.proto == IPPROTO_ICMP) {
+ const struct icmp *icmp = (const struct icmp *) data;
+ if (len < 8 || ntohs(icmp->icmp_id) != global_id)
+ return false;
+ reply->token = ntohs(icmp->icmp_seq);
+ } else if (hdr.version == 6 && hdr.proto == IPPROTO_ICMPV6) {
+ const struct icmpv6_msg_echo *echo = (struct icmpv6_msg_echo *) ((char *) data + sizeof(struct icmpv6_hdr));
+ if (len < 8 || ntohs(echo->icmpv6_id) != global_id)
+ return false;
+ reply->token = ntohs(echo->icmpv6_seq);
+ } else if (hdr.proto == IPPROTO_TCP) {
+ const struct tcp_hdr *tcp = (const struct tcp_hdr *) data;
+ if (len < 2)
+ return false;
+ reply->token = ntohs(tcp->th_sport) ^ global_id;
+ } else if (hdr.proto == IPPROTO_UDP) {
+ const struct udp_hdr *udp = (const struct udp_hdr *) data;
+ if (len < 2)
+ return false;
+ reply->token = ntohs(udp->uh_sport) ^ global_id;
+ } else if (hdr.proto == IPPROTO_SCTP) {
+ const struct sctp_hdr *sctp = (const struct sctp_hdr *) data;
+ if (len < 2)
+ return false;
+ reply->token = ntohs(sctp->sh_sport) ^ global_id;
+ } else {
+ if (len < 6)
+ return false;
+ /* Check IP ID for proto scan. */
+ reply->token = hdr.ipid ^ global_id;
+ }
+
+ reply->target_addr = hdr.dst;
+
+ return true;
+}
+
+static bool decode_reply(const void *ip, unsigned int len, Reply *reply) {
+ struct abstract_ip_hdr hdr;
+ const void *data;
+
+ data = ip_get_data(ip, &len, &hdr);
+ if (data == NULL)
+ return false;
+
+ reply->from_addr = hdr.src;
+ reply->ttl = hdr.ttl;
+
+ if (hdr.version == 4 && hdr.proto == IPPROTO_ICMP) {
+ /* ICMP responses comprise all the TTL exceeded messages we expect from all
+ probe types, as well as actual replies from ICMP probes. */
+ const struct icmp_hdr *icmp = (const struct icmp_hdr *) data;
+ if (len < 8)
+ return false;
+ if ((icmp->icmp_type == ICMP_TIMEXCEED
+ && icmp->icmp_code == ICMP_TIMEXCEED_INTRANS)
+ || icmp->icmp_type == ICMP_UNREACH) {
+ /* Get the encapsulated IP packet. */
+ const void *encaps = icmp_get_data(icmp, &len);
+ if (encaps == NULL)
+ return false;
+ return parse_encapsulated_reply(encaps, len, reply);
+ } else if (icmp->icmp_type == ICMP_ECHOREPLY
+ || icmp->icmp_type == ICMP_MASKREPLY
+ || icmp->icmp_type == ICMP_TSTAMPREPLY) {
+ /* Need this alternate form of header for icmp_id and icmp_seq. */
+ const struct icmp *icmp = (const struct icmp *) data;
+
+ if (ntohs(icmp->icmp_id) != global_id)
+ return false;
+ reply->token = ntohs(icmp->icmp_seq);
+ /* Reply came directly from the target. */
+ reply->target_addr = reply->from_addr;
+ } else {
+ return false;
+ }
+ } else if (hdr.version == 6 && hdr.proto == IP_PROTO_ICMPV6) {
+ /* ICMPv6 responses comprise all the TTL exceeded messages we expect from
+ all probe types, as well as actual replies from ICMP probes. */
+ const struct icmpv6_hdr *icmpv6 = (const struct icmpv6_hdr *) data;
+ if (len < 2)
+ return false;
+ /* TIMEXCEED, UNREACH */
+ if ((icmpv6->icmpv6_type == ICMPV6_TIMEXCEED
+ && icmpv6->icmpv6_code == ICMPV6_TIMEXCEED_INTRANS)
+ || icmpv6->icmpv6_type == ICMPV6_UNREACH) {
+ /* Get the encapsulated IP packet. */
+ const void *encaps = icmpv6_get_data(icmpv6, &len);
+ if (encaps == NULL)
+ return false;
+ return parse_encapsulated_reply(encaps, len, reply);
+ } else if (icmpv6->icmpv6_type == ICMPV6_ECHOREPLY) {
+ /* MASKREPLY, TSTAMPREPLY */
+ const struct icmpv6_msg_echo *echo;
+
+ if (len < sizeof(*icmpv6) + 4)
+ return false;
+ echo = (struct icmpv6_msg_echo *) ((char *) icmpv6 + sizeof(*icmpv6));
+ if (ntohs(echo->icmpv6_id) != global_id)
+ return false;
+ reply->token = ntohs(echo->icmpv6_seq);
+ /* Reply came directly from the target. */
+ reply->target_addr = reply->from_addr;
+ } else {
+ return false;
+ }
+ } else if (hdr.proto == IPPROTO_TCP) {
+ const struct tcp_hdr *tcp = (const struct tcp_hdr *) data;
+ if (len < 4)
+ return false;
+ reply->token = ntohs(tcp->th_dport) ^ global_id;
+ reply->target_addr = reply->from_addr;
+ } else if (hdr.proto == IPPROTO_UDP) {
+ const struct udp_hdr *udp = (const struct udp_hdr *) data;
+ if (len < 4)
+ return false;
+ reply->token = ntohs(udp->uh_dport) ^ global_id;
+ reply->target_addr = reply->from_addr;
+ } else if (hdr.proto == IPPROTO_SCTP) {
+ const struct sctp_hdr *sctp = (const struct sctp_hdr *) data;
+ if (len < 4)
+ return false;
+ reply->token = ntohs(sctp->sh_dport) ^ global_id;
+ reply->target_addr = reply->from_addr;
+ } else {
+ return false;
+ }
+
+ return true;
+}
+
+static bool read_reply(Reply *reply, pcap_t *pd, long timeout) {
+ const struct ip *ip;
+ unsigned int iplen;
+ struct link_header linkhdr;
+
+ ip = (struct ip *) readip_pcap(pd, &iplen, timeout, &reply->rcvdtime, &linkhdr, true);
+ if (ip == NULL)
+ return false;
+ if (ip->ip_v == 4 || ip->ip_v == 6)
+ return decode_reply(ip, iplen, reply);
+ else
+ return false;
+}
+
+void TracerouteState::read_replies(long timeout) {
+ struct timeval now;
+ Reply reply;
+
+ assert(timeout / 1000 <= (long) o.scan_delay);
+ timeout = MAX(timeout, 10000);
+ now = get_now();
+
+ while (timeout > 0 && read_reply(&reply, pd, timeout)) {
+ std::list<Probe *>::iterator it;
+ struct timeval oldnow;
+ HostState *host;
+ Probe *probe;
+ float rtt;
+
+ oldnow = now;
+ now = get_now();
+ timeout -= TIMEVAL_SUBTRACT(now, oldnow);
+
+ probe = this->lookup_probe(&reply.target_addr, reply.token);
+ if (probe == NULL)
+ continue;
+ host = probe->host;
+
+ if (sockaddr_storage_equal(host->target->TargetSockAddr(), &reply.from_addr)) {
+ adjust_timeouts2(&probe->sent_time, &reply.rcvdtime, &host->target->to);
+ if (host->reached_target == 0 || probe->ttl < host->reached_target)
+ host->reached_target = probe->ttl;
+ if (host->state == HostState::COUNTING_DOWN) {
+ /* If this probe was past the target, skip ahead to what we think the
+ actual distance is. */
+ int distance = get_initial_ttl_guess(reply.ttl) - reply.ttl + 1;
+ if (distance > 0 && distance < host->current_ttl)
+ host->current_ttl = MIN(distance, MAX_TTL);
+ }
+ num_active_probes -= host->cancel_probes_above(probe->ttl);
+ }
+
+ rtt = TIMEVAL_SUBTRACT(reply.rcvdtime, probe->sent_time) / 1000.0;
+ set_host_hop(host, probe->ttl, &reply.from_addr, rtt);
+
+ it = find(host->unanswered_probes.begin(), host->unanswered_probes.end(), probe);
+ num_active_probes -= host->cancel_probe(it);
+ }
+}
+
+void TracerouteState::cull_timeouts() {
+ std::list<HostState *>::iterator host_iter;
+ struct timeval now;
+
+ now = get_now();
+
+ for (host_iter = active_hosts.begin(); host_iter != active_hosts.end(); host_iter++) {
+ while (!(*host_iter)->active_probes.empty()
+ && (*host_iter)->active_probes.front()->is_timedout(&now)) {
+ Probe *probe;
+
+ probe = (*host_iter)->active_probes.front();
+ if (o.debugging > 1) {
+ log_write(LOG_STDOUT, "Traceroute probe to %s TTL %d timed out\n",
+ probe->host->target->targetipstr(), probe->ttl);
+ }
+ set_host_hop_timedout(*host_iter, probe->ttl);
+ (*host_iter)->active_probes.pop_front();
+ num_active_probes--;
+ if (probe->may_resend())
+ (*host_iter)->pending_resends.push_front(probe);
+ }
+ }
+}
+
+void TracerouteState::remove_finished_hosts() {
+ std::list<HostState *>::iterator it, next;
+
+ for (it = active_hosts.begin(); it != active_hosts.end(); it = next) {
+ next = it;
+ next++;
+ if ((*it)->is_finished()) {
+ if (next_sending_host == it)
+ next_active_host();
+ active_hosts.erase(it);
+ }
+ }
+}
+
+/* Dummy class to use sockaddr_storage as a map key. */
+struct lt_sockaddr_storage {
+ bool operator()(const struct sockaddr_storage& a, const struct sockaddr_storage& b) const {
+ return sockaddr_storage_cmp(&a, &b) < 0;
+ }
+};
+
+/* Find the reverse-DNS names of the hops. */
+void TracerouteState::resolve_hops() {
+ std::set<sockaddr_storage, lt_sockaddr_storage> addrs;
+ std::set<sockaddr_storage, lt_sockaddr_storage>::iterator addr_iter;
+ std::vector<HostState *>::iterator host_iter;
+ std::map<sockaddr_storage, const char *, lt_sockaddr_storage> name_map;
+ Target **targets;
+ Hop *hop;
+ int i, n;
+
+ /* First, put all the IPv4 addresses in a set to remove duplicates. This
+ re-resolves the addresses of the targets themselves, which is a little
+ inefficient. */
+ for (host_iter = hosts.begin(); host_iter != hosts.end(); host_iter++) {
+ for (hop = (*host_iter)->hops; hop != NULL; hop = hop->parent) {
+ if (hop->addr.ss_family != AF_UNSPEC)
+ addrs.insert(hop->addr);
+ }
+ }
+ n = addrs.size();
+ /* Second, make an array of pointer to Target to suit the interface of
+ nmap_mass_rdns. */
+ targets = (Target **) safe_malloc(sizeof(*targets) * n);
+ i = 0;
+ addr_iter = addrs.begin();
+ while (i < n) {
+ targets[i] = new Target();
+ targets[i]->setTargetSockAddr(&*addr_iter, sizeof(*addr_iter));
+ targets[i]->flags = HOST_UP;
+ i++;
+ addr_iter++;
+ }
+ nmap_mass_rdns(targets, n);
+ /* Third, make a map from addresses to names for easy lookup. */
+ for (i = 0; i < n; i++) {
+ struct sockaddr_storage ss;
+ size_t ss_len;
+ const char *hostname = targets[i]->HostName();
+ if (*hostname == '\0')
+ hostname = NULL;
+ ss_len = sizeof(ss);
+ targets[i]->TargetSockAddr(&ss, &ss_len);
+ name_map[ss] = hostname;
+ }
+ /* Finally, copy the names into the hops. */
+ for (host_iter = hosts.begin(); host_iter != hosts.end(); host_iter++) {
+ for (hop = (*host_iter)->hops; hop != NULL; hop = hop->parent) {
+ if (hop->addr.ss_family != AF_UNSPEC) {
+ const char *hostname = name_map[hop->addr];
+ if (hostname != NULL)
+ hop->hostname = hostname;
+ }
+ }
+ }
+ for (i = 0; i < n; i++)
+ delete targets[i];
+ free(targets);
+}
+
+void TracerouteState::transfer_hops() {
+ std::vector<HostState *>::iterator it;
+ Hop *p;
+
+ for (it = hosts.begin(); it != hosts.end(); it++) {
+ for (p = (*it)->hops; p != NULL; p = p->parent) {
+ TracerouteHop hop;
+
+ /* Trim excessive hops. */
+ if ((*it)->reached_target && p->ttl > (*it)->reached_target)
+ continue;
+
+ hop.tag = p->tag;
+ if (p->addr.ss_family == 0) {
+ hop.timedout = true;
+ } else {
+ hop.timedout = false;
+ hop.rtt = p->rtt;
+ }
+ hop.name = p->hostname;
+ hop.addr = p->addr;
+ hop.ttl = p->ttl;
+ (*it)->target->traceroute_hops.push_front(hop);
+ }
+
+ (*it)->target->traceroute_probespec = (*it)->pspec;
+
+ /* Set the hop distance for OS fingerprints. */
+ if ((*it)->reached_target) {
+ (*it)->target->distance = (*it)->reached_target;
+ (*it)->target->distance_calculation_method = DIST_METHOD_TRACEROUTE;
+ }
+ }
+}
+
+Probe *TracerouteState::lookup_probe(
+ const struct sockaddr_storage *target_addr, u16 token) {
+ std::list<HostState *>::const_iterator host_iter;
+ std::list<Probe *>::iterator probe_iter;
+
+ for (host_iter = active_hosts.begin(); host_iter != active_hosts.end(); host_iter++) {
+
+ if (!sockaddr_storage_equal((*host_iter)->target->TargetSockAddr(), target_addr))
+ continue;
+ for (probe_iter = (*host_iter)->unanswered_probes.begin();
+ probe_iter != (*host_iter)->unanswered_probes.end();
+ probe_iter++) {
+ if ((*probe_iter)->token == token)
+ return *probe_iter;
+ }
+ }
+
+ return NULL;
+}
+
+double TracerouteState::completion_fraction() const {
+ std::vector<HostState *>::const_iterator it;
+ double sum;
+
+ sum = 0.0;
+ for (it = hosts.begin(); it != hosts.end(); it++)
+ sum += (*it)->completion_fraction();
+ return sum / hosts.size();
+}
+
+/* This is a special case of traceroute when all the targets are directly
+ connected. Because the distance to each target is known to be 1, we send no
+ probes at all, only fill in a TracerouteHop structure. */
+static int traceroute_direct(std::vector<Target *> targets) {
+ std::vector<Target *>::iterator it;
+
+ for (it = targets.begin(); it != targets.end(); it++) {
+ TracerouteHop hop;
+ const char *hostname;
+ size_t sslen;
+
+ sslen = sizeof(hop.tag);
+ (*it)->TargetSockAddr(&hop.tag, &sslen);
+ hop.timedout = false;
+ hop.rtt = (*it)->to.srtt / 1000.0;
+ hostname = (*it)->HostName();
+ if (hostname != NULL && hostname[0] != '\0')
+ hop.name = hostname;
+ hop.addr = hop.tag;
+ hop.ttl = 1;
+ (*it)->traceroute_hops.push_front(hop);
+ }
+
+ return 1;
+}
+
+static int traceroute_remote(std::vector<Target *> targets) {
+ std::vector<Target *>::iterator target_iter;
+
+ if (targets.empty())
+ return 1;
+
+ if (timedout_hops == NULL) {
+ timedout_hops = new std::list<Hop *>;
+ }
+
+ TracerouteState global_state(targets);
+
+ global_id = get_random_u16();
+
+ ScanProgressMeter SPM("Traceroute");
+
+ o.current_scantype = TRACEROUTE;
+
+ while (!global_state.active_hosts.empty()) {
+ struct timeval now;
+ long int timeout;
+
+ global_state.send_new_probes();
+ now = get_now();
+ timeout = TIMEVAL_SUBTRACT(global_state.next_send_time, now);
+ global_state.read_replies(timeout);
+ global_state.cull_timeouts();
+ global_state.remove_finished_hosts();
+
+ if (keyWasPressed())
+ SPM.printStats(global_state.completion_fraction(), NULL);
+ }
+
+ SPM.endTask(NULL, NULL);
+
+ if (!o.noresolve)
+ global_state.resolve_hops();
+ /* This puts the hops into the targets known by the global_state. */
+ global_state.transfer_hops();
+
+ /* Update initial_ttl to be the highest distance seen in this host group, as
+ an estimate for the next. */
+ initial_ttl = 0;
+ for (target_iter = targets.begin();
+ target_iter != targets.end();
+ target_iter++) {
+ initial_ttl = MAX(initial_ttl, (*target_iter)->traceroute_hops.size());
+ }
+
+ if (hop_cache_size() > MAX_HOP_CACHE_SIZE) {
+ if (o.debugging) {
+ log_write(LOG_STDOUT, "Clearing hop cache that has grown to %d\n",
+ hop_cache_size());
+ }
+ traceroute_hop_cache_clear();
+ }
+
+ return 1;
+}
+
+int traceroute(std::vector<Target *> &Targets) {
+ std::vector<Target *> direct, remote;
+ std::vector<Target *>::iterator target_iter;
+
+ /* Separate directly connected targets from remote targets. */
+ for (target_iter = Targets.begin();
+ target_iter != Targets.end();
+ target_iter++) {
+ if ((*target_iter)->ifType() == devt_loopback)
+ ; /* Ignore */
+ else if ((*target_iter)->directlyConnected())
+ direct.push_back(*target_iter);
+ else
+ remote.push_back(*target_iter);
+ }
+
+ traceroute_direct(direct);
+ traceroute_remote(remote);
+
+ return 1;
+}
+
+static struct timeval get_now(struct timeval *now) {
+ struct timeval tv;
+
+ if (now != NULL)
+ return *now;
+ gettimeofday(&tv, NULL);
+
+ return tv;
+}
+
+/* Convert the address in ss to a string. The result is returned in a static
+ buffer so you can't call this twice in arguments to printf, for example. */
+static const char *ss_to_string(const struct sockaddr_storage *ss) {
+ return inet_ntop_ez(ss, sizeof(*ss));
+}