summaryrefslogtreecommitdiff
path: root/osdep/ManagedRoute.hpp
blob: 9c7e847749a8b5191864e734f3e861fd306c1fcc (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
#ifndef ZT_MANAGEDROUTE_HPP
#define ZT_MANAGEDROUTE_HPP

#include <stdlib.h>
#include <string.h>

#include "../node/InetAddress.hpp"
#include "../node/Utils.hpp"

#include <stdexcept>
#include <vector>
#include <set>

namespace ZeroTier {

/**
 * A ZT-managed route that used C++ RAII semantics to automatically clean itself up on deallocate
 */
class ManagedRoute
{
public:
	ManagedRoute()
	{
		_device[0] = (char)0;
		_systemDevice[0] = (char)0;
	}

	~ManagedRoute()
	{
		this->remove();
	}

	ManagedRoute(const ManagedRoute &r)
	{
		*this = r;
	}

	inline ManagedRoute &operator=(const ManagedRoute &r)
	{
		if (_applied.size() == 0) {
			_target = r._target;
			_via = r._via;
			_systemVia = r._systemVia;
			_applied = r._applied;
			Utils::scopy(_device,sizeof(_device),r._device);
			Utils::scopy(_systemDevice,sizeof(_systemDevice),r._systemDevice);
		} else {
			// Sanity check -- this is an 'assert' to detect a bug
			fprintf(stderr,"Applied ManagedRoute isn't copyable!\n");
			abort();
		}
		return *this;
	}

	/**
	 * Initialize object and set route
	 *
	 * Note: on Windows, use the interface NET_LUID in hexadecimal as the
	 * "device name."
	 *
	 * @param target Route target (e.g. 0.0.0.0/0 for default)
	 * @param via Route next L3 hop or NULL InetAddress if local in which case it will be routed via device
	 * @param device Name or hex LUID of ZeroTier device (e.g. zt#)
	 * @return True if route was successfully set
	 */
	inline bool set(const InetAddress &target,const InetAddress &via,const char *device)
	{
		if ((!via)&&(!device[0]))
			return false;
		this->remove();
		_target = target;
		_via = via;
		if (via.ss_family == AF_INET)
			_via.setPort(32);
		else if (via.ss_family == AF_INET6)
			_via.setPort(128);
		Utils::scopy(_device,sizeof(_device),device);
		return this->sync();
	}

	/**
	 * Set or update currently set route
	 *
	 * This must be called periodically for routes that shadow others so that
	 * shadow routes can be updated. In some cases it has no effect
	 *
	 * @return True if route add/update was successful
	 */
	bool sync();

	/**
	 * Remove and clear this ManagedRoute
	 *
	 * This does nothing if this ManagedRoute is not set or has already been
	 * removed. If this is not explicitly called it is called automatically on
	 * destruct.
	 */
	void remove();

	inline const InetAddress &target() const { return _target; }
	inline const InetAddress &via() const { return _via; }
	inline const char *device() const { return _device; }

private:
	InetAddress _target;
	InetAddress _via;
	InetAddress _systemVia; // for route overrides
	std::set<InetAddress> _applied; // routes currently applied
	char _device[128];
	char _systemDevice[128]; // for route overrides
};

} // namespace ZeroTier

#endif