1 /* Copyright (c) 2001 Matej Pfajfar.
2 * Copyright (c) 2001-2004, Roger Dingledine.
3 * Copyright (c) 2004-2006, Roger Dingledine, Nick Mathewson.
4 * Copyright (c) 2007-2021, The Tor Project, Inc. */
5 /* See LICENSE for licensing information */
8 * \file connection_edge.h
9 * \brief Header file for connection_edge.c.
12 #ifndef TOR_CONNECTION_EDGE_H
13 #define TOR_CONNECTION_EDGE_H
15 #include "lib/testsupport/testsupport.h"
16 #include "lib/encoding/confline.h"
18 #include "feature/hs/hs_service.h"
20 edge_connection_t
*TO_EDGE_CONN(connection_t
*);
21 entry_connection_t
*TO_ENTRY_CONN(connection_t
*);
22 entry_connection_t
*EDGE_TO_ENTRY_CONN(edge_connection_t
*);
24 const edge_connection_t
*CONST_TO_EDGE_CONN(const connection_t
*);
25 const entry_connection_t
*CONST_TO_ENTRY_CONN(const connection_t
*);
26 const entry_connection_t
*CONST_EDGE_TO_ENTRY_CONN(const edge_connection_t
*);
28 #define EXIT_CONN_STATE_MIN_ 1
29 /** State for an exit connection: waiting for response from DNS farm. */
30 #define EXIT_CONN_STATE_RESOLVING 1
31 /** State for an exit connection: waiting for connect() to finish. */
32 #define EXIT_CONN_STATE_CONNECTING 2
33 /** State for an exit connection: open and ready to transmit data. */
34 #define EXIT_CONN_STATE_OPEN 3
35 /** State for an exit connection: waiting to be removed. */
36 #define EXIT_CONN_STATE_RESOLVEFAILED 4
37 #define EXIT_CONN_STATE_MAX_ 4
39 /* The AP state values must be disjoint from the EXIT state values. */
40 #define AP_CONN_STATE_MIN_ 5
41 /** State for a SOCKS connection: waiting for SOCKS request. */
42 #define AP_CONN_STATE_SOCKS_WAIT 5
43 /** State for a SOCKS connection: got a y.onion URL; waiting to receive
44 * rendezvous descriptor. */
45 #define AP_CONN_STATE_RENDDESC_WAIT 6
46 /** The controller will attach this connection to a circuit; it isn't our
48 #define AP_CONN_STATE_CONTROLLER_WAIT 7
49 /** State for a SOCKS connection: waiting for a completed circuit. */
50 #define AP_CONN_STATE_CIRCUIT_WAIT 8
51 /** State for a SOCKS connection: sent BEGIN, waiting for CONNECTED. */
52 #define AP_CONN_STATE_CONNECT_WAIT 9
53 /** State for a SOCKS connection: sent RESOLVE, waiting for RESOLVED. */
54 #define AP_CONN_STATE_RESOLVE_WAIT 10
55 /** State for a SOCKS connection: ready to send and receive. */
56 #define AP_CONN_STATE_OPEN 11
57 /** State for a transparent natd connection: waiting for original
59 #define AP_CONN_STATE_NATD_WAIT 12
60 /** State for an HTTP tunnel: waiting for an HTTP CONNECT command. */
61 #define AP_CONN_STATE_HTTP_CONNECT_WAIT 13
62 #define AP_CONN_STATE_MAX_ 13
64 #define EXIT_PURPOSE_MIN_ 1
65 /** This exit stream wants to do an ordinary connect. */
66 #define EXIT_PURPOSE_CONNECT 1
67 /** This exit stream wants to do a resolve (either normal or reverse). */
68 #define EXIT_PURPOSE_RESOLVE 2
69 #define EXIT_PURPOSE_MAX_ 2
71 /** True iff the AP_CONN_STATE_* value <b>s</b> means that the corresponding
72 * edge connection is not attached to any circuit. */
73 #define AP_CONN_STATE_IS_UNATTACHED(s) \
74 ((s) <= AP_CONN_STATE_CIRCUIT_WAIT || (s) == AP_CONN_STATE_NATD_WAIT)
76 #define connection_mark_unattached_ap(conn, endreason) \
77 connection_mark_unattached_ap_((conn), (endreason), __LINE__, SHORT_FILE__)
79 /** Possible return values for parse_extended_hostname. */
80 typedef enum hostname_type_t
{
87 MOCK_DECL(void,connection_mark_unattached_ap_
,
88 (entry_connection_t
*conn
, int endreason
,
89 int line
, const char *file
));
90 int connection_edge_reached_eof(edge_connection_t
*conn
);
91 int connection_edge_process_inbuf(edge_connection_t
*conn
,
93 int connection_edge_destroy(circid_t circ_id
, edge_connection_t
*conn
);
94 int connection_edge_end(edge_connection_t
*conn
, uint8_t reason
);
95 int connection_edge_end_errno(edge_connection_t
*conn
);
96 void connection_edge_end_close(edge_connection_t
*conn
, uint8_t reason
);
97 int connection_edge_flushed_some(edge_connection_t
*conn
);
98 int connection_edge_finished_flushing(edge_connection_t
*conn
);
99 int connection_edge_finished_connecting(edge_connection_t
*conn
);
101 void connection_entry_set_controller_wait(entry_connection_t
*conn
);
103 void connection_ap_about_to_close(entry_connection_t
*edge_conn
);
104 void connection_exit_about_to_close(edge_connection_t
*edge_conn
);
105 void connection_reapply_exit_policy(config_line_t
*changes
);
108 connection_ap_handshake_send_begin
,(entry_connection_t
*ap_conn
));
109 int connection_ap_handshake_send_resolve(entry_connection_t
*ap_conn
);
111 entry_connection_t
*connection_ap_make_link(connection_t
*partner
,
112 char *address
, uint16_t port
,
116 int use_begindir
, int want_onehop
);
117 void connection_ap_handshake_socks_reply(entry_connection_t
*conn
, char *reply
,
120 MOCK_DECL(void,connection_ap_handshake_socks_resolved
,
121 (entry_connection_t
*conn
,
124 const uint8_t *answer
,
127 void connection_ap_handshake_socks_resolved_addr(entry_connection_t
*conn
,
128 const tor_addr_t
*answer
,
132 int connection_exit_begin_conn(cell_t
*cell
, circuit_t
*circ
);
133 int connection_exit_begin_resolve(cell_t
*cell
, or_circuit_t
*circ
);
134 void connection_exit_connect(edge_connection_t
*conn
);
135 int connection_edge_is_rendezvous_stream(const edge_connection_t
*conn
);
136 int connection_ap_can_use_exit(const entry_connection_t
*conn
,
138 void connection_ap_expire_beginning(void);
139 void connection_ap_rescan_and_attach_pending(void);
140 void connection_ap_attach_pending(int retry
);
141 void connection_ap_mark_as_pending_circuit_(entry_connection_t
*entry_conn
,
142 const char *file
, int line
);
143 #define connection_ap_mark_as_pending_circuit(c) \
144 connection_ap_mark_as_pending_circuit_((c), __FILE__, __LINE__)
145 void connection_ap_mark_as_non_pending_circuit(entry_connection_t
*entry_conn
);
146 void connection_ap_mark_as_waiting_for_renddesc(
147 entry_connection_t
*entry_conn
);
149 #define CONNECTION_AP_EXPECT_NONPENDING(c) do { \
150 if (ENTRY_TO_CONN(c)->state == AP_CONN_STATE_CIRCUIT_WAIT) { \
151 log_warn(LD_BUG, "At %s:%d: %p was unexpectedly in circuit_wait.", \
152 __FILE__, __LINE__, (c)); \
153 connection_ap_mark_as_non_pending_circuit(c); \
156 void connection_ap_fail_onehop(const char *failed_digest
,
157 cpath_build_state_t
*build_state
);
158 void circuit_discard_optional_exit_enclaves(extend_info_t
*info
);
159 int connection_ap_detach_retriable(entry_connection_t
*conn
,
160 origin_circuit_t
*circ
,
162 int connection_ap_process_transparent(entry_connection_t
*conn
);
164 int address_is_invalid_destination(const char *address
, int client
);
166 MOCK_DECL(int, connection_ap_rewrite_and_attach_if_allowed
,
167 (entry_connection_t
*conn
,
168 origin_circuit_t
*circ
,
169 crypt_path_t
*cpath
));
170 int connection_ap_handshake_rewrite_and_attach(entry_connection_t
*conn
,
171 origin_circuit_t
*circ
,
172 crypt_path_t
*cpath
);
174 #if defined(HAVE_NET_IF_H) && defined(HAVE_NET_PFVAR_H)
175 int get_pf_socket(void);
178 int connection_edge_compatible_with_circuit(const entry_connection_t
*conn
,
179 const origin_circuit_t
*circ
);
180 int connection_edge_update_circuit_isolation(const entry_connection_t
*conn
,
181 origin_circuit_t
*circ
,
183 void circuit_clear_isolation(origin_circuit_t
*circ
);
184 streamid_t
get_unique_stream_id_by_circ(origin_circuit_t
*circ
);
186 void connection_edge_free_all(void);
188 void connection_ap_warn_and_unmark_if_pending_circ(
189 entry_connection_t
*entry_conn
,
192 /** Lowest value for DNS ttl clipping excluding the random addition. */
193 #define MIN_DNS_TTL (5*60)
194 /** Highest value for DNS ttl clipping excluding the random addition. */
195 #define MAX_DNS_TTL (60*60)
196 /** How long do we keep DNS cache entries before purging them (regardless of
198 #define MAX_DNS_ENTRY_AGE (3*60*60)
199 /** How long do we cache/tell clients to cache DNS records when no TTL is
201 #define DEFAULT_DNS_TTL (30*60)
202 /** How much should we +- each TTL to make it fuzzy with uniform sampling at
203 * exits? The value 4 minutes was chosen so that the lowest possible clip is
204 * 60s. Such low clips were used in the past for all TTLs due to a bug in Tor,
205 * see "The effect of DNS on Tor's Anonymity" by Greschbach et al. In other
206 * words, sampling such low clips is unlikely to cause any breakage at exits.
208 #define FUZZY_DNS_TTL (4*60)
210 uint32_t clip_dns_ttl(uint32_t ttl
);
211 uint32_t clip_dns_fuzzy_ttl(uint32_t ttl
);
213 int connection_half_edge_is_valid_data(const smartlist_t
*half_conns
,
214 streamid_t stream_id
);
215 int connection_half_edge_is_valid_sendme(const smartlist_t
*half_conns
,
216 streamid_t stream_id
);
217 int connection_half_edge_is_valid_connected(const smartlist_t
*half_conns
,
218 streamid_t stream_id
);
219 int connection_half_edge_is_valid_end(smartlist_t
*half_conns
,
220 streamid_t stream_id
);
221 int connection_half_edge_is_valid_resolved(smartlist_t
*half_conns
,
222 streamid_t stream_id
);
223 bool connection_half_edges_waiting(const origin_circuit_t
*circ
);
225 size_t half_streams_get_total_allocation(void);
227 void half_edge_free_(struct half_edge_t
*he
);
228 #define half_edge_free(he) \
229 FREE_AND_NULL(half_edge_t, half_edge_free_, (he))
231 /** @name Begin-cell flags
233 * These flags are used in RELAY_BEGIN cells to change the default behavior
238 /** When this flag is set, the client is willing to get connected to IPv6
240 #define BEGIN_FLAG_IPV6_OK (1u<<0)
241 /** When this flag is set, the client DOES NOT support connecting to IPv4
242 * addresses. (The sense of this flag is inverted from IPV6_OK, so that the
243 * old default behavior of Tor is equivalent to having all flags set to 0.)
245 #define BEGIN_FLAG_IPV4_NOT_OK (1u<<1)
246 /** When this flag is set, if we find both an IPv4 and an IPv6 address,
247 * we use the IPv6 address. Otherwise we use the IPv4 address. */
248 #define BEGIN_FLAG_IPV6_PREFERRED (1u<<2)
251 #ifdef CONNECTION_EDGE_PRIVATE
253 STATIC
bool parse_extended_hostname(char *address
, hostname_type_t
*type_out
);
255 /** A parsed BEGIN or BEGIN_DIR cell */
256 typedef struct begin_cell_t
{
257 /** The address the client has asked us to connect to, or NULL if this is
260 /** The flags specified in the BEGIN cell's body. One or more of
263 /** The client's requested port. */
265 /** The client's requested Stream ID */
267 /** True iff this is a BEGIN_DIR cell. */
268 unsigned is_begindir
: 1;
271 STATIC
int begin_cell_parse(const cell_t
*cell
, begin_cell_t
*bcell
,
272 uint8_t *end_reason_out
);
273 STATIC
int connected_cell_format_payload(uint8_t *payload_out
,
274 const tor_addr_t
*addr
,
278 /** Original address, after we lowercased it but before we started
281 char orig_address
[MAX_SOCKS_ADDR_LEN
];
282 /** True iff the address has been automatically remapped to a local
283 * address in VirtualAddrNetwork. (Only set true when we do a resolve
284 * and get a virtual address; not when we connect to the address.) */
286 /** If this connection has a .exit address, who put it there? */
287 addressmap_entry_source_t exit_source
;
288 /** If we've rewritten the address, when does this map expire? */
290 /** If we should close the connection, this is the end_reason to pass
291 * to connection_mark_unattached_ap */
293 /** True iff we should close the connection, either because of error or
294 * because of successful early RESOLVED reply. */
298 STATIC
void connection_ap_handshake_rewrite(entry_connection_t
*conn
,
299 rewrite_result_t
*out
);
301 STATIC
int connection_ap_process_http_connect(entry_connection_t
*conn
);
302 STATIC
void export_hs_client_circuit_id(edge_connection_t
*edge_conn
,
303 hs_circuit_id_protocol_t protocol
);
306 STATIC
void connection_half_edge_add(const edge_connection_t
*conn
,
307 origin_circuit_t
*circ
);
308 STATIC
struct half_edge_t
*connection_half_edge_find_stream_id(
309 const smartlist_t
*half_conns
,
310 streamid_t stream_id
);
311 #endif /* defined(CONNECTION_EDGE_PRIVATE) */
313 #endif /* !defined(TOR_CONNECTION_EDGE_H) */