xref: /freebsd/sys/netinet/libalias/alias_db.h (revision 4fbb9c43aa44d9145151bb5f77d302ba01fb7551)
1 /*-
2  * SPDX-License-Identifier: BSD-2-Clause
3  *
4  * Copyright (c) 2001 Charles Mott <cm@linktel.net>
5  * All rights reserved.
6  *
7  * Redistribution and use in source and binary forms, with or without
8  * modification, are permitted provided that the following conditions
9  * are met:
10  * 1. Redistributions of source code must retain the above copyright
11  *    notice, this list of conditions and the following disclaimer.
12  * 2. Redistributions in binary form must reproduce the above copyright
13  *    notice, this list of conditions and the following disclaimer in the
14  *    documentation and/or other materials provided with the distribution.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
17  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
20  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26  * SUCH DAMAGE.
27  */
28 
29 /*
30     Alias_db.c encapsulates all data structures used for storing
31     packet aliasing data.  Other parts of the aliasing software
32     access data through functions provided in this file.
33 
34     Data storage is based on the notion of a "link", which is
35     established for ICMP echo/reply packets, UDP datagrams and
36     TCP stream connections.  A link stores the original source
37     and destination addresses.  For UDP and TCP, it also stores
38     source and destination port numbers, as well as an alias
39     port number.  Links are also used to store information about
40     fragments.
41 
42     There is a facility for sweeping through and deleting old
43     links as new packets are sent through.  A simple timeout is
44     used for ICMP and UDP links.  TCP links are left alone unless
45     there is an incomplete connection, in which case the link
46     can be deleted after a certain amount of time.
47 
48     Initial version: August, 1996  (cjm)
49 
50     Version 1.4: September 16, 1996 (cjm)
51 	Facility for handling incoming links added.
52 
53     Version 1.6: September 18, 1996 (cjm)
54 	ICMP data handling simplified.
55 
56     Version 1.7: January 9, 1997 (cjm)
57 	Fragment handling simplified.
58 	Saves pointers for unresolved fragments.
59 	Permits links for unspecified remote ports
60 	  or unspecified remote addresses.
61 	Fixed bug which did not properly zero port
62 	  table entries after a link was deleted.
63 	Cleaned up some obsolete comments.
64 
65     Version 1.8: January 14, 1997 (cjm)
66 	Fixed data type error in StartPoint().
67 	(This error did not exist prior to v1.7
68 	and was discovered and fixed by Ari Suutari)
69 
70     Version 1.9: February 1, 1997
71 	Optionally, connections initiated from packet aliasing host
72 	machine will will not have their port number aliased unless it
73 	conflicts with an aliasing port already being used. (cjm)
74 
75 	All options earlier being #ifdef'ed are now available through
76 	a new interface, SetPacketAliasMode().  This allows run time
77 	control (which is now available in PPP+pktAlias through the
78 	'alias' keyword). (ee)
79 
80 	Added ability to create an alias port without
81 	either destination address or port specified.
82 	port type = ALIAS_PORT_UNKNOWN_DEST_ALL (ee)
83 
84 	Removed K&R style function headers
85 	and general cleanup. (ee)
86 
87 	Added packetAliasMode to replace compiler #defines's (ee)
88 
89 	Allocates sockets for partially specified
90 	ports if ALIAS_USE_SOCKETS defined. (cjm)
91 
92     Version 2.0: March, 1997
93 	SetAliasAddress() will now clean up alias links
94 	if the aliasing address is changed. (cjm)
95 
96 	PacketAliasPermanentLink() function added to support permanent
97 	links.  (J. Fortes suggested the need for this.)
98 	Examples:
99 
100 	(192.168.0.1, port 23)  <-> alias port 6002, unknown dest addr/port
101 
102 	(192.168.0.2, port 21)  <-> alias port 3604, known dest addr
103 						     unknown dest port
104 
105 	These permanent links allow for incoming connections to
106 	machines on the local network.  They can be given with a
107 	user-chosen amount of specificity, with increasing specificity
108 	meaning more security. (cjm)
109 
110 	Quite a bit of rework to the basic engine.  The portTable[]
111 	array, which kept track of which ports were in use was replaced
112 	by a table/linked list structure. (cjm)
113 
114 	SetExpire() function added. (cjm)
115 
116 	DeleteLink() no longer frees memory association with a pointer
117 	to a fragment (this bug was first recognized by E. Eklund in
118 	v1.9).
119 
120     Version 2.1: May, 1997 (cjm)
121 	Packet aliasing engine reworked so that it can handle
122 	multiple external addresses rather than just a single
123 	host address.
124 
125 	PacketAliasRedirectPort() and PacketAliasRedirectAddr()
126 	added to the API.  The first function is a more generalized
127 	version of PacketAliasPermanentLink().  The second function
128 	implements static network address translation.
129 
130     Version 3.2: July, 2000 (salander and satoh)
131 	Added FindNewPortGroup to get contiguous range of port values.
132 
133 	Added QueryUdpTcpIn and QueryUdpTcpOut to look for an aliasing
134 	link but not actually add one.
135 
136 	Added FindRtspOut, which is closely derived from FindUdpTcpOut,
137 	except that the alias port (from FindNewPortGroup) is provided
138 	as input.
139 
140     See HISTORY file for additional revisions.
141 */
142 
143 #ifndef _ALIAS_DB_H_
144 #define _ALIAS_DB_H_
145 
146 
147 /*
148    Constants (note: constants are also defined
149 	      near relevant functions or structs)
150 */
151 
152 /* Timeouts (in seconds) for different link types */
153 #define ICMP_EXPIRE_TIME             60
154 #define UDP_EXPIRE_TIME              60
155 #define PROTO_EXPIRE_TIME            60
156 #define FRAGMENT_ID_EXPIRE_TIME      10
157 #define FRAGMENT_PTR_EXPIRE_TIME     30
158 
159 /* TCP link expire time for different cases */
160 /* When the link has been used and closed - minimal grace time to
161    allow ACKs and potential re-connect in FTP (XXX - is this allowed?)  */
162 #ifndef TCP_EXPIRE_DEAD
163 #define TCP_EXPIRE_DEAD           10
164 #endif
165 
166 /* When the link has been used and closed on one side - the other side
167    is allowed to still send data */
168 #ifndef TCP_EXPIRE_SINGLEDEAD
169 #define TCP_EXPIRE_SINGLEDEAD     90
170 #endif
171 
172 /* When the link isn't yet up */
173 #ifndef TCP_EXPIRE_INITIAL
174 #define TCP_EXPIRE_INITIAL       300
175 #endif
176 
177 /* When the link is up */
178 #ifndef TCP_EXPIRE_CONNECTED
179 #define TCP_EXPIRE_CONNECTED   86400
180 #endif
181 
182 /* Dummy port number codes used for FindLinkIn/Out() and AddLink().
183    These constants can be anything except zero, which indicates an
184    unknown port number. */
185 
186 #define NO_DEST_PORT     1
187 #define NO_SRC_PORT      1
188 
189 /* Matches any/unknown address in FindLinkIn/Out() and AddLink(). */
190 static struct in_addr const ANY_ADDR = { INADDR_ANY };
191 
192 /* Data Structures
193 
194     The fundamental data structure used in this program is
195     "struct alias_link".  Whenever a TCP connection is made,
196     a UDP datagram is sent out, or an ICMP echo request is made,
197     a link record is made (if it has not already been created).
198     The link record is identified by the source address/port
199     and the destination address/port. In the case of an ICMP
200     echo request, the source port is treated as being equivalent
201     with the 16-bit ID number of the ICMP packet.
202 
203     The link record also can store some auxiliary data.  For
204     TCP connections that have had sequence and acknowledgment
205     modifications, data space is available to track these changes.
206     A state field is used to keep track in changes to the TCP
207     connection state.  ID numbers of fragments can also be
208     stored in the auxiliary space.  Pointers to unresolved
209     fragments can also be stored.
210 
211     The link records support two independent chainings.  Lookup
212     tables for input and out tables hold the initial pointers
213     the link chains.  On input, the lookup table indexes on alias
214     port and link type.  On output, the lookup table indexes on
215     source address, destination address, source port, destination
216     port and link type.
217 */
218 
219 /* used to save changes to ACK/sequence numbers */
220 struct ack_data_record {
221 	u_long		ack_old;
222 	u_long		ack_new;
223 	int		delta;
224 	int		active;
225 };
226 
227 /* Information about TCP connection */
228 struct tcp_state {
229 	int		in;	/* State for outside -> inside */
230 	int		out;	/* State for inside  -> outside */
231 	int		index;	/* Index to ACK data array */
232 	/* Indicates whether ACK and sequence numbers been modified */
233 	int		ack_modified;
234 };
235 
236 /* Number of distinct ACK number changes
237  * saved for a modified TCP stream */
238 #define N_LINK_TCP_DATA   3
239 struct tcp_dat {
240 	struct tcp_state state;
241 	struct ack_data_record ack[N_LINK_TCP_DATA];
242 	/* Which firewall record is used for this hole? */
243 	int		fwhole;
244 };
245 
246 /* LSNAT server pool (circular list) */
247 struct server {
248 	struct in_addr	addr;
249 	u_short		port;
250 	struct server  *next;
251 };
252 
253 /* Main data structure */
254 struct alias_link {
255 	struct libalias *la;
256 	/* Address and port information */
257 	struct in_addr	src_addr;
258 	struct in_addr	dst_addr;
259 	struct in_addr	alias_addr;
260 	struct in_addr	proxy_addr;
261 	u_short		src_port;
262 	u_short		dst_port;
263 	u_short		alias_port;
264 	u_short		proxy_port;
265 	struct server  *server;
266 	/* Type of link: TCP, UDP, ICMP, proto, frag */
267 	int		link_type;
268 /* values for link_type */
269 #define LINK_ICMP                     IPPROTO_ICMP
270 #define LINK_UDP                      IPPROTO_UDP
271 #define LINK_TCP                      IPPROTO_TCP
272 #define LINK_FRAGMENT_ID              (IPPROTO_MAX + 1)
273 #define LINK_FRAGMENT_PTR             (IPPROTO_MAX + 2)
274 #define LINK_ADDR                     (IPPROTO_MAX + 3)
275 #define LINK_PPTP                     (IPPROTO_MAX + 4)
276 
277 	int		flags;	/* indicates special characteristics */
278 	int		pflags;	/* protocol-specific flags */
279 /* flag bits */
280 #define LINK_UNKNOWN_DEST_PORT     0x01
281 #define LINK_UNKNOWN_DEST_ADDR     0x02
282 #define LINK_PERMANENT             0x04
283 #define LINK_PARTIALLY_SPECIFIED   0x03	/* logical-or of first two bits */
284 #define LINK_UNFIREWALLED          0x08
285 
286 	int		timestamp;	/* Time link was last accessed */
287 #ifndef NO_USE_SOCKETS
288 	int		sockfd;		/* socket descriptor */
289 #endif
290 	/* Linked list of pointers for input and output lookup tables  */
291 	union {
292 		struct {
293 			SPLAY_ENTRY(alias_link) out;
294 			LIST_ENTRY (alias_link) in;
295 		} all;
296 		struct {
297 			LIST_ENTRY (alias_link) list;
298 		} pptp;
299 	};
300 	struct {
301 		TAILQ_ENTRY(alias_link) list;
302 		int	time;	/* Expire time for link */
303 	} expire;
304 	/* Auxiliary data */
305 	union {
306 		char           *frag_ptr;
307 		struct in_addr	frag_addr;
308 		struct tcp_dat *tcp;
309 	} data;
310 };
311 
312 /* Clean up procedure. */
313 static void finishoff(void);
314 
315 /* Internal utility routines (used only in alias_db.c)
316 
317 Lookup table starting points:
318     StartPointIn()           -- link table initial search point for
319 				incoming packets
320     StartPointOut()          -- link table initial search point for
321 				outgoing packets
322 
323 Miscellaneous:
324     SeqDiff()                -- difference between two TCP sequences
325     ShowAliasStats()         -- send alias statistics to a monitor file
326 */
327 
328 /* Local prototypes */
329 static struct group_in *
330 StartPointIn(struct libalias *, struct in_addr, u_short, int, int);
331 static int	SeqDiff(u_long, u_long);
332 
333 #ifndef NO_FW_PUNCH
334 /* Firewall control */
335 static void	InitPunchFW(struct libalias *);
336 static void	UninitPunchFW(struct libalias *);
337 static void	ClearFWHole(struct alias_link *);
338 
339 #endif
340 
341 /* Log file control */
342 static void	ShowAliasStats(struct libalias *);
343 static int	InitPacketAliasLog(struct libalias *);
344 static void	UninitPacketAliasLog(struct libalias *);
345 
346 void		SctpShowAliasStats(struct libalias *la);
347 
348 
349 /* Splay handling */
350 static inline int
351 cmp_out(struct alias_link *a, struct alias_link *b) {
352 	int i = a->src_port - b->src_port;
353 	if (i != 0) return (i);
354 	if (a->src_addr.s_addr > b->src_addr.s_addr) return (1);
355 	if (a->src_addr.s_addr < b->src_addr.s_addr) return (-1);
356 	if (a->dst_addr.s_addr > b->dst_addr.s_addr) return (1);
357 	if (a->dst_addr.s_addr < b->dst_addr.s_addr) return (-1);
358 	i = a->dst_port - b->dst_port;
359 	if (i != 0) return (i);
360 	i = a->link_type - b->link_type;
361 	return (i);
362 }
363 SPLAY_PROTOTYPE(splay_out, alias_link, all.out, cmp_out);
364 
365 static inline int
366 cmp_in(struct group_in *a, struct group_in *b) {
367 	int i = a->alias_port - b->alias_port;
368 	if (i != 0) return (i);
369 	i = a->link_type - b->link_type;
370 	if (i != 0) return (i);
371 	if (a->alias_addr.s_addr > b->alias_addr.s_addr) return (1);
372 	if (a->alias_addr.s_addr < b->alias_addr.s_addr) return (-1);
373 	return (0);
374 }
375 SPLAY_PROTOTYPE(splay_in, group_in, in, cmp_in);
376 
377 /* Internal routines for finding, deleting and adding links
378 
379 Port Allocation:
380     GetNewPort()             -- find and reserve new alias port number
381     GetSocket()              -- try to allocate a socket for a given port
382 
383 Link creation and deletion:
384     CleanupAliasData()      - remove all link chains from lookup table
385     CleanupLink()           - look for a stale link
386     DeleteLink()            - remove link
387     AddLink()               - add link
388     ReLink()                - change link
389 
390 Link search:
391     FindLinkOut()           - find link for outgoing packets
392     FindLinkIn()            - find link for incoming packets
393 
394 Port search:
395     FindNewPortGroup()      - find an available group of ports
396 */
397 
398 /* Local prototypes */
399 static int	GetNewPort(struct libalias *, struct alias_link *, int);
400 #ifndef NO_USE_SOCKETS
401 static u_short	GetSocket(struct libalias *, u_short, int *, int);
402 #endif
403 static void	CleanupAliasData(struct libalias *, int);
404 static void	CleanupLink(struct libalias *, struct alias_link **, int);
405 static void	DeleteLink(struct alias_link **, int);
406 static struct alias_link *
407 UseLink(struct libalias *, struct alias_link *);
408 
409 static struct alias_link *
410 ReLink(struct alias_link *,
411     struct in_addr, struct in_addr, struct in_addr,
412     u_short, u_short, int, int, int);
413 
414 static struct alias_link *
415 FindLinkOut(struct libalias *, struct in_addr, struct in_addr, u_short, u_short, int, int);
416 
417 static struct alias_link *
418 FindLinkIn(struct libalias *, struct in_addr, struct in_addr, u_short, u_short, int, int);
419 
420 static u_short _RandomPort(struct libalias *la);
421 
422 #define GET_NEW_PORT_MAX_ATTEMPTS       20
423 
424 
425 #ifndef NO_FW_PUNCH
426 
427 static void ClearAllFWHoles(struct libalias *la);
428 
429 #define fw_setfield(la, field, num)			\
430 do {						\
431     (field)[(num) - la->fireWallBaseNum] = 1;		\
432 } /*lint -save -e717 */ while(0)/* lint -restore */
433 
434 #define fw_clrfield(la, field, num)			\
435 do {							\
436     (field)[(num) - la->fireWallBaseNum] = 0;		\
437 } /*lint -save -e717 */ while(0)/* lint -restore */
438 
439 #define fw_tstfield(la, field, num) ((field)[(num) - la->fireWallBaseNum])
440 
441 #endif /* !NO_FW_PUNCH */
442 
443 #endif /* _ALIAS_DB_H_ */
444