xref: /freebsd/sys/netinet/libalias/alias_db.c (revision 3110d4ebd6c0848cf5e25890d01791bb407e2a9b)
1 /*-
2  * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
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 #include <sys/cdefs.h>
30 __FBSDID("$FreeBSD$");
31 
32 /*
33     Alias_db.c encapsulates all data structures used for storing
34     packet aliasing data.  Other parts of the aliasing software
35     access data through functions provided in this file.
36 
37     Data storage is based on the notion of a "link", which is
38     established for ICMP echo/reply packets, UDP datagrams and
39     TCP stream connections.  A link stores the original source
40     and destination addresses.  For UDP and TCP, it also stores
41     source and destination port numbers, as well as an alias
42     port number.  Links are also used to store information about
43     fragments.
44 
45     There is a facility for sweeping through and deleting old
46     links as new packets are sent through.  A simple timeout is
47     used for ICMP and UDP links.  TCP links are left alone unless
48     there is an incomplete connection, in which case the link
49     can be deleted after a certain amount of time.
50 
51     Initial version: August, 1996  (cjm)
52 
53     Version 1.4: September 16, 1996 (cjm)
54 	Facility for handling incoming links added.
55 
56     Version 1.6: September 18, 1996 (cjm)
57 	ICMP data handling simplified.
58 
59     Version 1.7: January 9, 1997 (cjm)
60 	Fragment handling simplified.
61 	Saves pointers for unresolved fragments.
62 	Permits links for unspecified remote ports
63 	  or unspecified remote addresses.
64 	Fixed bug which did not properly zero port
65 	  table entries after a link was deleted.
66 	Cleaned up some obsolete comments.
67 
68     Version 1.8: January 14, 1997 (cjm)
69 	Fixed data type error in StartPoint().
70 	(This error did not exist prior to v1.7
71 	and was discovered and fixed by Ari Suutari)
72 
73     Version 1.9: February 1, 1997
74 	Optionally, connections initiated from packet aliasing host
75 	machine will will not have their port number aliased unless it
76 	conflicts with an aliasing port already being used. (cjm)
77 
78 	All options earlier being #ifdef'ed are now available through
79 	a new interface, SetPacketAliasMode().  This allows run time
80 	control (which is now available in PPP+pktAlias through the
81 	'alias' keyword). (ee)
82 
83 	Added ability to create an alias port without
84 	either destination address or port specified.
85 	port type = ALIAS_PORT_UNKNOWN_DEST_ALL (ee)
86 
87 	Removed K&R style function headers
88 	and general cleanup. (ee)
89 
90 	Added packetAliasMode to replace compiler #defines's (ee)
91 
92 	Allocates sockets for partially specified
93 	ports if ALIAS_USE_SOCKETS defined. (cjm)
94 
95     Version 2.0: March, 1997
96 	SetAliasAddress() will now clean up alias links
97 	if the aliasing address is changed. (cjm)
98 
99 	PacketAliasPermanentLink() function added to support permanent
100 	links.  (J. Fortes suggested the need for this.)
101 	Examples:
102 
103 	(192.168.0.1, port 23)  <-> alias port 6002, unknown dest addr/port
104 
105 	(192.168.0.2, port 21)  <-> alias port 3604, known dest addr
106 						     unknown dest port
107 
108 	These permanent links allow for incoming connections to
109 	machines on the local network.  They can be given with a
110 	user-chosen amount of specificity, with increasing specificity
111 	meaning more security. (cjm)
112 
113 	Quite a bit of rework to the basic engine.  The portTable[]
114 	array, which kept track of which ports were in use was replaced
115 	by a table/linked list structure. (cjm)
116 
117 	SetExpire() function added. (cjm)
118 
119 	DeleteLink() no longer frees memory association with a pointer
120 	to a fragment (this bug was first recognized by E. Eklund in
121 	v1.9).
122 
123     Version 2.1: May, 1997 (cjm)
124 	Packet aliasing engine reworked so that it can handle
125 	multiple external addresses rather than just a single
126 	host address.
127 
128 	PacketAliasRedirectPort() and PacketAliasRedirectAddr()
129 	added to the API.  The first function is a more generalized
130 	version of PacketAliasPermanentLink().  The second function
131 	implements static network address translation.
132 
133     Version 3.2: July, 2000 (salander and satoh)
134 	Added FindNewPortGroup to get contiguous range of port values.
135 
136 	Added QueryUdpTcpIn and QueryUdpTcpOut to look for an aliasing
137 	link but not actually add one.
138 
139 	Added FindRtspOut, which is closely derived from FindUdpTcpOut,
140 	except that the alias port (from FindNewPortGroup) is provided
141 	as input.
142 
143     See HISTORY file for additional revisions.
144 */
145 
146 #ifdef _KERNEL
147 #include <machine/stdarg.h>
148 #include <sys/param.h>
149 #include <sys/kernel.h>
150 #include <sys/systm.h>
151 #include <sys/lock.h>
152 #include <sys/module.h>
153 #include <sys/rwlock.h>
154 #include <sys/syslog.h>
155 #else
156 #include <stdarg.h>
157 #include <stdlib.h>
158 #include <stdio.h>
159 #include <sys/errno.h>
160 #include <sys/time.h>
161 #include <unistd.h>
162 #endif
163 
164 #include <sys/socket.h>
165 #include <netinet/tcp.h>
166 
167 #ifdef _KERNEL
168 #include <netinet/libalias/alias.h>
169 #include <netinet/libalias/alias_local.h>
170 #include <netinet/libalias/alias_mod.h>
171 #include <net/if.h>
172 #else
173 #include "alias.h"
174 #include "alias_local.h"
175 #include "alias_mod.h"
176 #endif
177 
178 static		LIST_HEAD(, libalias) instancehead = LIST_HEAD_INITIALIZER(instancehead);
179 
180 /*
181    Constants (note: constants are also defined
182 	      near relevant functions or structs)
183 */
184 
185 /* Parameters used for cleanup of expired links */
186 /* NOTE: ALIAS_CLEANUP_INTERVAL_SECS must be less then LINK_TABLE_OUT_SIZE */
187 #define ALIAS_CLEANUP_INTERVAL_SECS  64
188 #define ALIAS_CLEANUP_MAX_SPOKES     (LINK_TABLE_OUT_SIZE/5)
189 
190 /* Timeouts (in seconds) for different link types */
191 #define ICMP_EXPIRE_TIME             60
192 #define UDP_EXPIRE_TIME              60
193 #define PROTO_EXPIRE_TIME            60
194 #define FRAGMENT_ID_EXPIRE_TIME      10
195 #define FRAGMENT_PTR_EXPIRE_TIME     30
196 
197 /* TCP link expire time for different cases */
198 /* When the link has been used and closed - minimal grace time to
199    allow ACKs and potential re-connect in FTP (XXX - is this allowed?)  */
200 #ifndef TCP_EXPIRE_DEAD
201 #define TCP_EXPIRE_DEAD           10
202 #endif
203 
204 /* When the link has been used and closed on one side - the other side
205    is allowed to still send data */
206 #ifndef TCP_EXPIRE_SINGLEDEAD
207 #define TCP_EXPIRE_SINGLEDEAD     90
208 #endif
209 
210 /* When the link isn't yet up */
211 #ifndef TCP_EXPIRE_INITIAL
212 #define TCP_EXPIRE_INITIAL       300
213 #endif
214 
215 /* When the link is up */
216 #ifndef TCP_EXPIRE_CONNECTED
217 #define TCP_EXPIRE_CONNECTED   86400
218 #endif
219 
220 /* Dummy port number codes used for FindLinkIn/Out() and AddLink().
221    These constants can be anything except zero, which indicates an
222    unknown port number. */
223 
224 #define NO_DEST_PORT     1
225 #define NO_SRC_PORT      1
226 
227 /* Data Structures
228 
229     The fundamental data structure used in this program is
230     "struct alias_link".  Whenever a TCP connection is made,
231     a UDP datagram is sent out, or an ICMP echo request is made,
232     a link record is made (if it has not already been created).
233     The link record is identified by the source address/port
234     and the destination address/port. In the case of an ICMP
235     echo request, the source port is treated as being equivalent
236     with the 16-bit ID number of the ICMP packet.
237 
238     The link record also can store some auxiliary data.  For
239     TCP connections that have had sequence and acknowledgment
240     modifications, data space is available to track these changes.
241     A state field is used to keep track in changes to the TCP
242     connection state.  ID numbers of fragments can also be
243     stored in the auxiliary space.  Pointers to unresolved
244     fragments can also be stored.
245 
246     The link records support two independent chainings.  Lookup
247     tables for input and out tables hold the initial pointers
248     the link chains.  On input, the lookup table indexes on alias
249     port and link type.  On output, the lookup table indexes on
250     source address, destination address, source port, destination
251     port and link type.
252 */
253 
254 struct ack_data_record {	/* used to save changes to ACK/sequence
255 				 * numbers */
256 	u_long		ack_old;
257 	u_long		ack_new;
258 	int		delta;
259 	int		active;
260 };
261 
262 struct tcp_state {		/* Information about TCP connection        */
263 	int		in;	/* State for outside -> inside             */
264 	int		out;	/* State for inside  -> outside            */
265 	int		index;	/* Index to ACK data array                 */
266 	int		ack_modified;	/* Indicates whether ACK and
267 					 * sequence numbers */
268 	/* been modified                           */
269 };
270 
271 #define N_LINK_TCP_DATA   3	/* Number of distinct ACK number changes
272 				 * saved for a modified TCP stream */
273 struct tcp_dat {
274 	struct tcp_state state;
275 	struct ack_data_record ack[N_LINK_TCP_DATA];
276 	int		fwhole;	/* Which firewall record is used for this
277 				 * hole? */
278 };
279 
280 struct server {			/* LSNAT server pool (circular list) */
281 	struct in_addr	addr;
282 	u_short		port;
283 	struct server  *next;
284 };
285 
286 struct alias_link {		/* Main data structure */
287 	struct libalias *la;
288 	struct in_addr	src_addr;	/* Address and port information        */
289 	struct in_addr	dst_addr;
290 	struct in_addr	alias_addr;
291 	struct in_addr	proxy_addr;
292 	u_short		src_port;
293 	u_short		dst_port;
294 	u_short		alias_port;
295 	u_short		proxy_port;
296 	struct server  *server;
297 
298 	int		link_type;	/* Type of link: TCP, UDP, ICMP,
299 					 * proto, frag */
300 
301 /* values for link_type */
302 #define LINK_ICMP                     IPPROTO_ICMP
303 #define LINK_UDP                      IPPROTO_UDP
304 #define LINK_TCP                      IPPROTO_TCP
305 #define LINK_FRAGMENT_ID              (IPPROTO_MAX + 1)
306 #define LINK_FRAGMENT_PTR             (IPPROTO_MAX + 2)
307 #define LINK_ADDR                     (IPPROTO_MAX + 3)
308 #define LINK_PPTP                     (IPPROTO_MAX + 4)
309 
310 	int		flags;	/* indicates special characteristics   */
311 	int		pflags;	/* protocol-specific flags */
312 
313 /* flag bits */
314 #define LINK_UNKNOWN_DEST_PORT     0x01
315 #define LINK_UNKNOWN_DEST_ADDR     0x02
316 #define LINK_PERMANENT             0x04
317 #define LINK_PARTIALLY_SPECIFIED   0x03	/* logical-or of first two bits */
318 #define LINK_UNFIREWALLED          0x08
319 
320 	int		timestamp;	/* Time link was last accessed         */
321 	int		expire_time;	/* Expire time for link                */
322 #ifndef	NO_USE_SOCKETS
323 	int		sockfd;	/* socket descriptor                   */
324 #endif
325 			LIST_ENTRY    (alias_link) list_out;	/* Linked list of
326 								 * pointers for     */
327 			LIST_ENTRY    (alias_link) list_in;	/* input and output
328 								 * lookup tables  */
329 
330 	union {			/* Auxiliary data                      */
331 		char           *frag_ptr;
332 		struct in_addr	frag_addr;
333 		struct tcp_dat *tcp;
334 	}		data;
335 };
336 
337 /* Clean up procedure. */
338 static void finishoff(void);
339 
340 /* Kernel module definition. */
341 #ifdef	_KERNEL
342 MALLOC_DEFINE(M_ALIAS, "libalias", "packet aliasing");
343 
344 MODULE_VERSION(libalias, 1);
345 
346 static int
347 alias_mod_handler(module_t mod, int type, void *data)
348 {
349 
350 	switch (type) {
351 	case MOD_QUIESCE:
352 	case MOD_UNLOAD:
353 	        finishoff();
354 	case MOD_LOAD:
355 		return (0);
356 	default:
357 		return (EINVAL);
358 	}
359 }
360 
361 static moduledata_t alias_mod = {
362        "alias", alias_mod_handler, NULL
363 };
364 
365 DECLARE_MODULE(alias, alias_mod, SI_SUB_DRIVERS, SI_ORDER_SECOND);
366 #endif
367 
368 /* Internal utility routines (used only in alias_db.c)
369 
370 Lookup table starting points:
371     StartPointIn()           -- link table initial search point for
372 				incoming packets
373     StartPointOut()          -- link table initial search point for
374 				outgoing packets
375 
376 Miscellaneous:
377     SeqDiff()                -- difference between two TCP sequences
378     ShowAliasStats()         -- send alias statistics to a monitor file
379 */
380 
381 /* Local prototypes */
382 static u_int	StartPointIn(struct in_addr, u_short, int);
383 
384 static		u_int
385 StartPointOut(struct in_addr, struct in_addr,
386     u_short, u_short, int);
387 
388 static int	SeqDiff(u_long, u_long);
389 
390 #ifndef NO_FW_PUNCH
391 /* Firewall control */
392 static void	InitPunchFW(struct libalias *);
393 static void	UninitPunchFW(struct libalias *);
394 static void	ClearFWHole(struct alias_link *);
395 
396 #endif
397 
398 /* Log file control */
399 static void	ShowAliasStats(struct libalias *);
400 static int	InitPacketAliasLog(struct libalias *);
401 static void	UninitPacketAliasLog(struct libalias *);
402 
403 void SctpShowAliasStats(struct libalias *la);
404 
405 static		u_int
406 StartPointIn(struct in_addr alias_addr,
407     u_short alias_port,
408     int link_type)
409 {
410 	u_int n;
411 
412 	n = alias_addr.s_addr;
413 	if (link_type != LINK_PPTP)
414 		n += alias_port;
415 	n += link_type;
416 	return (n % LINK_TABLE_IN_SIZE);
417 }
418 
419 static		u_int
420 StartPointOut(struct in_addr src_addr, struct in_addr dst_addr,
421     u_short src_port, u_short dst_port, int link_type)
422 {
423 	u_int n;
424 
425 	n = src_addr.s_addr;
426 	n += dst_addr.s_addr;
427 	if (link_type != LINK_PPTP) {
428 		n += src_port;
429 		n += dst_port;
430 	}
431 	n += link_type;
432 
433 	return (n % LINK_TABLE_OUT_SIZE);
434 }
435 
436 static int
437 SeqDiff(u_long x, u_long y)
438 {
439 /* Return the difference between two TCP sequence numbers */
440 
441 /*
442     This function is encapsulated in case there are any unusual
443     arithmetic conditions that need to be considered.
444 */
445 
446 	return (ntohl(y) - ntohl(x));
447 }
448 
449 #ifdef _KERNEL
450 
451 static void
452 AliasLog(char *str, const char *format, ...)
453 {
454 	va_list ap;
455 
456 	va_start(ap, format);
457 	vsnprintf(str, LIBALIAS_BUF_SIZE, format, ap);
458 	va_end(ap);
459 }
460 #else
461 static void
462 AliasLog(FILE *stream, const char *format, ...)
463 {
464 	va_list ap;
465 
466 	va_start(ap, format);
467 	vfprintf(stream, format, ap);
468 	va_end(ap);
469 	fflush(stream);
470 }
471 #endif
472 
473 static void
474 ShowAliasStats(struct libalias *la)
475 {
476 
477 	LIBALIAS_LOCK_ASSERT(la);
478 /* Used for debugging */
479 	if (la->logDesc) {
480 		int tot  = la->icmpLinkCount + la->udpLinkCount +
481 		  (la->sctpLinkCount>>1) + /* sctp counts half associations */
482 			la->tcpLinkCount + la->pptpLinkCount +
483 			la->protoLinkCount + la->fragmentIdLinkCount +
484 			la->fragmentPtrLinkCount;
485 
486 		AliasLog(la->logDesc,
487 			 "icmp=%u, udp=%u, tcp=%u, sctp=%u, pptp=%u, proto=%u, frag_id=%u frag_ptr=%u / tot=%u",
488 			 la->icmpLinkCount,
489 			 la->udpLinkCount,
490 			 la->tcpLinkCount,
491 			 la->sctpLinkCount>>1, /* sctp counts half associations */
492 			 la->pptpLinkCount,
493 			 la->protoLinkCount,
494 			 la->fragmentIdLinkCount,
495 			 la->fragmentPtrLinkCount, tot);
496 #ifndef _KERNEL
497 		AliasLog(la->logDesc, " (sock=%u)\n", la->sockCount);
498 #endif
499 	}
500 }
501 
502 void SctpShowAliasStats(struct libalias *la)
503 {
504 
505 	ShowAliasStats(la);
506 }
507 
508 /* Internal routines for finding, deleting and adding links
509 
510 Port Allocation:
511     GetNewPort()             -- find and reserve new alias port number
512     GetSocket()              -- try to allocate a socket for a given port
513 
514 Link creation and deletion:
515     CleanupAliasData()      - remove all link chains from lookup table
516     IncrementalCleanup()    - look for stale links in a single chain
517     DeleteLink()            - remove link
518     AddLink()               - add link
519     ReLink()                - change link
520 
521 Link search:
522     FindLinkOut()           - find link for outgoing packets
523     FindLinkIn()            - find link for incoming packets
524 
525 Port search:
526     FindNewPortGroup()      - find an available group of ports
527 */
528 
529 /* Local prototypes */
530 static int	GetNewPort(struct libalias *, struct alias_link *, int);
531 #ifndef	NO_USE_SOCKETS
532 static u_short	GetSocket(struct libalias *, u_short, int *, int);
533 #endif
534 static void	CleanupAliasData(struct libalias *);
535 
536 static void	IncrementalCleanup(struct libalias *);
537 
538 static void	DeleteLink(struct alias_link *);
539 
540 static struct alias_link *
541 ReLink(struct alias_link *,
542     struct in_addr, struct in_addr, struct in_addr,
543     u_short, u_short, int, int);
544 
545 static struct alias_link *
546 		FindLinkOut   (struct libalias *, struct in_addr, struct in_addr, u_short, u_short, int, int);
547 
548 static struct alias_link *
549 		FindLinkIn    (struct libalias *, struct in_addr, struct in_addr, u_short, u_short, int, int);
550 
551 #define ALIAS_PORT_BASE            0x08000
552 #define ALIAS_PORT_MASK            0x07fff
553 #define ALIAS_PORT_MASK_EVEN       0x07ffe
554 #define GET_NEW_PORT_MAX_ATTEMPTS       20
555 
556 #define FIND_EVEN_ALIAS_BASE             1
557 
558 /* GetNewPort() allocates port numbers.  Note that if a port number
559    is already in use, that does not mean that it cannot be used by
560    another link concurrently.  This is because GetNewPort() looks for
561    unused triplets: (dest addr, dest port, alias port). */
562 
563 static int
564 GetNewPort(struct libalias *la, struct alias_link *lnk, int alias_port_param)
565 {
566 	int i;
567 	int max_trials;
568 	u_short port_sys;
569 	u_short port_net;
570 
571 	LIBALIAS_LOCK_ASSERT(la);
572 /*
573    Description of alias_port_param for GetNewPort().  When
574    this parameter is zero or positive, it precisely specifies
575    the port number.  GetNewPort() will return this number
576    without check that it is in use.
577 
578    When this parameter is GET_ALIAS_PORT, it indicates to get a randomly
579    selected port number.
580 */
581 
582 	if (alias_port_param == GET_ALIAS_PORT) {
583 		/*
584 		 * The aliasing port is automatically selected by one of
585 		 * two methods below:
586 		 */
587 		max_trials = GET_NEW_PORT_MAX_ATTEMPTS;
588 
589 		if (la->packetAliasMode & PKT_ALIAS_SAME_PORTS) {
590 			/*
591 			 * When the PKT_ALIAS_SAME_PORTS option is chosen,
592 			 * the first try will be the actual source port. If
593 			 * this is already in use, the remainder of the
594 			 * trials will be random.
595 			 */
596 			port_net = lnk->src_port;
597 			port_sys = ntohs(port_net);
598 		} else {
599 			/* First trial and all subsequent are random. */
600 			port_sys = arc4random() & ALIAS_PORT_MASK;
601 			port_sys += ALIAS_PORT_BASE;
602 			port_net = htons(port_sys);
603 		}
604 	} else if (alias_port_param >= 0 && alias_port_param < 0x10000) {
605 		lnk->alias_port = (u_short) alias_port_param;
606 		return (0);
607 	} else {
608 #ifdef LIBALIAS_DEBUG
609 		fprintf(stderr, "PacketAlias/GetNewPort(): ");
610 		fprintf(stderr, "input parameter error\n");
611 #endif
612 		return (-1);
613 	}
614 
615 /* Port number search */
616 	for (i = 0; i < max_trials; i++) {
617 		int go_ahead;
618 		struct alias_link *search_result;
619 
620 		search_result = FindLinkIn(la, lnk->dst_addr, lnk->alias_addr,
621 		    lnk->dst_port, port_net,
622 		    lnk->link_type, 0);
623 
624 		if (search_result == NULL)
625 			go_ahead = 1;
626 		else if (!(lnk->flags & LINK_PARTIALLY_SPECIFIED)
627 		    && (search_result->flags & LINK_PARTIALLY_SPECIFIED))
628 			go_ahead = 1;
629 		else
630 			go_ahead = 0;
631 
632 		if (go_ahead) {
633 #ifndef	NO_USE_SOCKETS
634 			if ((la->packetAliasMode & PKT_ALIAS_USE_SOCKETS)
635 			    && (lnk->flags & LINK_PARTIALLY_SPECIFIED)
636 			    && ((lnk->link_type == LINK_TCP) ||
637 			    (lnk->link_type == LINK_UDP))) {
638 				if (GetSocket(la, port_net, &lnk->sockfd, lnk->link_type)) {
639 					lnk->alias_port = port_net;
640 					return (0);
641 				}
642 			} else {
643 #endif
644 				lnk->alias_port = port_net;
645 				return (0);
646 #ifndef	NO_USE_SOCKETS
647 			}
648 #endif
649 		}
650 		port_sys = arc4random() & ALIAS_PORT_MASK;
651 		port_sys += ALIAS_PORT_BASE;
652 		port_net = htons(port_sys);
653 	}
654 
655 #ifdef LIBALIAS_DEBUG
656 	fprintf(stderr, "PacketAlias/GetNewPort(): ");
657 	fprintf(stderr, "could not find free port\n");
658 #endif
659 
660 	return (-1);
661 }
662 
663 #ifndef	NO_USE_SOCKETS
664 static		u_short
665 GetSocket(struct libalias *la, u_short port_net, int *sockfd, int link_type)
666 {
667 	int err;
668 	int sock;
669 	struct sockaddr_in sock_addr;
670 
671 	LIBALIAS_LOCK_ASSERT(la);
672 	if (link_type == LINK_TCP)
673 		sock = socket(AF_INET, SOCK_STREAM, 0);
674 	else if (link_type == LINK_UDP)
675 		sock = socket(AF_INET, SOCK_DGRAM, 0);
676 	else {
677 #ifdef LIBALIAS_DEBUG
678 		fprintf(stderr, "PacketAlias/GetSocket(): ");
679 		fprintf(stderr, "incorrect link type\n");
680 #endif
681 		return (0);
682 	}
683 
684 	if (sock < 0) {
685 #ifdef LIBALIAS_DEBUG
686 		fprintf(stderr, "PacketAlias/GetSocket(): ");
687 		fprintf(stderr, "socket() error %d\n", *sockfd);
688 #endif
689 		return (0);
690 	}
691 	sock_addr.sin_family = AF_INET;
692 	sock_addr.sin_addr.s_addr = htonl(INADDR_ANY);
693 	sock_addr.sin_port = port_net;
694 
695 	err = bind(sock,
696 	    (struct sockaddr *)&sock_addr,
697 	    sizeof(sock_addr));
698 	if (err == 0) {
699 		la->sockCount++;
700 		*sockfd = sock;
701 		return (1);
702 	} else {
703 		close(sock);
704 		return (0);
705 	}
706 }
707 #endif
708 
709 /* FindNewPortGroup() returns a base port number for an available
710    range of contiguous port numbers. Note that if a port number
711    is already in use, that does not mean that it cannot be used by
712    another link concurrently.  This is because FindNewPortGroup()
713    looks for unused triplets: (dest addr, dest port, alias port). */
714 
715 int
716 FindNewPortGroup(struct libalias *la,
717     struct in_addr dst_addr,
718     struct in_addr alias_addr,
719     u_short src_port,
720     u_short dst_port,
721     u_short port_count,
722     u_char proto,
723     u_char align)
724 {
725 	int i, j;
726 	int max_trials;
727 	u_short port_sys;
728 	int link_type;
729 
730 	LIBALIAS_LOCK_ASSERT(la);
731 	/*
732 	 * Get link_type from protocol
733 	 */
734 
735 	switch (proto) {
736 	case IPPROTO_UDP:
737 		link_type = LINK_UDP;
738 		break;
739 	case IPPROTO_TCP:
740 		link_type = LINK_TCP;
741 		break;
742 	default:
743 		return (0);
744 		break;
745 	}
746 
747 	/*
748 	 * The aliasing port is automatically selected by one of two
749 	 * methods below:
750 	 */
751 	max_trials = GET_NEW_PORT_MAX_ATTEMPTS;
752 
753 	if (la->packetAliasMode & PKT_ALIAS_SAME_PORTS) {
754 		/*
755 		 * When the ALIAS_SAME_PORTS option is chosen, the first
756 		 * try will be the actual source port. If this is already
757 		 * in use, the remainder of the trials will be random.
758 		 */
759 		port_sys = ntohs(src_port);
760 
761 	} else {
762 		/* First trial and all subsequent are random. */
763 		if (align == FIND_EVEN_ALIAS_BASE)
764 			port_sys = arc4random() & ALIAS_PORT_MASK_EVEN;
765 		else
766 			port_sys = arc4random() & ALIAS_PORT_MASK;
767 
768 		port_sys += ALIAS_PORT_BASE;
769 	}
770 
771 /* Port number search */
772 	for (i = 0; i < max_trials; i++) {
773 		struct alias_link *search_result;
774 
775 		for (j = 0; j < port_count; j++)
776 			if ((search_result = FindLinkIn(la, dst_addr,
777 			    alias_addr, dst_port, htons(port_sys + j),
778 			    link_type, 0)) != NULL)
779 				break;
780 
781 		/* Found a good range, return base */
782 		if (j == port_count)
783 			return (htons(port_sys));
784 
785 		/* Find a new base to try */
786 		if (align == FIND_EVEN_ALIAS_BASE)
787 			port_sys = arc4random() & ALIAS_PORT_MASK_EVEN;
788 		else
789 			port_sys = arc4random() & ALIAS_PORT_MASK;
790 
791 		port_sys += ALIAS_PORT_BASE;
792 	}
793 
794 #ifdef LIBALIAS_DEBUG
795 	fprintf(stderr, "PacketAlias/FindNewPortGroup(): ");
796 	fprintf(stderr, "could not find free port(s)\n");
797 #endif
798 
799 	return (0);
800 }
801 
802 static void
803 CleanupAliasData(struct libalias *la)
804 {
805 	struct alias_link *lnk;
806 	int i;
807 
808 	LIBALIAS_LOCK_ASSERT(la);
809 	for (i = 0; i < LINK_TABLE_OUT_SIZE; i++) {
810 		lnk = LIST_FIRST(&la->linkTableOut[i]);
811 		while (lnk != NULL) {
812 			struct alias_link *link_next = LIST_NEXT(lnk, list_out);
813 			DeleteLink(lnk);
814 			lnk = link_next;
815 		}
816 	}
817 
818 	la->cleanupIndex = 0;
819 }
820 
821 static void
822 IncrementalCleanup(struct libalias *la)
823 {
824 	struct alias_link *lnk, *lnk_tmp;
825 
826 	LIBALIAS_LOCK_ASSERT(la);
827 	LIST_FOREACH_SAFE(lnk, &la->linkTableOut[la->cleanupIndex++],
828 	    list_out, lnk_tmp) {
829 		if (la->timeStamp - lnk->timestamp > lnk->expire_time)
830 			DeleteLink(lnk);
831 	}
832 
833 	if (la->cleanupIndex == LINK_TABLE_OUT_SIZE)
834 		la->cleanupIndex = 0;
835 }
836 
837 static void
838 DeleteLink(struct alias_link *lnk)
839 {
840 	struct libalias *la = lnk->la;
841 
842 	LIBALIAS_LOCK_ASSERT(la);
843 /* Don't do anything if the link is marked permanent */
844 	if (la->deleteAllLinks == 0 && lnk->flags & LINK_PERMANENT)
845 		return;
846 
847 #ifndef NO_FW_PUNCH
848 /* Delete associated firewall hole, if any */
849 	ClearFWHole(lnk);
850 #endif
851 
852 /* Free memory allocated for LSNAT server pool */
853 	if (lnk->server != NULL) {
854 		struct server *head, *curr, *next;
855 
856 		head = curr = lnk->server;
857 		do {
858 			next = curr->next;
859 			free(curr);
860 		} while ((curr = next) != head);
861 	}
862 /* Adjust output table pointers */
863 	LIST_REMOVE(lnk, list_out);
864 
865 /* Adjust input table pointers */
866 	LIST_REMOVE(lnk, list_in);
867 #ifndef	NO_USE_SOCKETS
868 /* Close socket, if one has been allocated */
869 	if (lnk->sockfd != -1) {
870 		la->sockCount--;
871 		close(lnk->sockfd);
872 	}
873 #endif
874 /* Link-type dependent cleanup */
875 	switch (lnk->link_type) {
876 	case LINK_ICMP:
877 		la->icmpLinkCount--;
878 		break;
879 	case LINK_UDP:
880 		la->udpLinkCount--;
881 		break;
882 	case LINK_TCP:
883 		la->tcpLinkCount--;
884 		free(lnk->data.tcp);
885 		break;
886 	case LINK_PPTP:
887 		la->pptpLinkCount--;
888 		break;
889 	case LINK_FRAGMENT_ID:
890 		la->fragmentIdLinkCount--;
891 		break;
892 	case LINK_FRAGMENT_PTR:
893 		la->fragmentPtrLinkCount--;
894 		if (lnk->data.frag_ptr != NULL)
895 			free(lnk->data.frag_ptr);
896 		break;
897 	case LINK_ADDR:
898 		break;
899 	default:
900 		la->protoLinkCount--;
901 		break;
902 	}
903 
904 /* Free memory */
905 	free(lnk);
906 
907 /* Write statistics, if logging enabled */
908 	if (la->packetAliasMode & PKT_ALIAS_LOG) {
909 		ShowAliasStats(la);
910 	}
911 }
912 
913 struct alias_link *
914 AddLink(struct libalias *la, struct in_addr src_addr, struct in_addr dst_addr,
915     struct in_addr alias_addr, u_short src_port, u_short dst_port,
916     int alias_port_param, int link_type)
917 {
918 	u_int start_point;
919 	struct alias_link *lnk;
920 
921 	LIBALIAS_LOCK_ASSERT(la);
922 	lnk = malloc(sizeof(struct alias_link));
923 	if (lnk != NULL) {
924 		/* Basic initialization */
925 		lnk->la = la;
926 		lnk->src_addr = src_addr;
927 		lnk->dst_addr = dst_addr;
928 		lnk->alias_addr = alias_addr;
929 		lnk->proxy_addr.s_addr = INADDR_ANY;
930 		lnk->src_port = src_port;
931 		lnk->dst_port = dst_port;
932 		lnk->proxy_port = 0;
933 		lnk->server = NULL;
934 		lnk->link_type = link_type;
935 #ifndef	NO_USE_SOCKETS
936 		lnk->sockfd = -1;
937 #endif
938 		lnk->flags = 0;
939 		lnk->pflags = 0;
940 		lnk->timestamp = la->timeStamp;
941 
942 		/* Expiration time */
943 		switch (link_type) {
944 		case LINK_ICMP:
945 			lnk->expire_time = ICMP_EXPIRE_TIME;
946 			break;
947 		case LINK_UDP:
948 			lnk->expire_time = UDP_EXPIRE_TIME;
949 			break;
950 		case LINK_TCP:
951 			lnk->expire_time = TCP_EXPIRE_INITIAL;
952 			break;
953 		case LINK_PPTP:
954 			lnk->flags |= LINK_PERMANENT;	/* no timeout. */
955 			break;
956 		case LINK_FRAGMENT_ID:
957 			lnk->expire_time = FRAGMENT_ID_EXPIRE_TIME;
958 			break;
959 		case LINK_FRAGMENT_PTR:
960 			lnk->expire_time = FRAGMENT_PTR_EXPIRE_TIME;
961 			break;
962 		case LINK_ADDR:
963 			break;
964 		default:
965 			lnk->expire_time = PROTO_EXPIRE_TIME;
966 			break;
967 		}
968 
969 		/* Determine alias flags */
970 		if (dst_addr.s_addr == INADDR_ANY)
971 			lnk->flags |= LINK_UNKNOWN_DEST_ADDR;
972 		if (dst_port == 0)
973 			lnk->flags |= LINK_UNKNOWN_DEST_PORT;
974 
975 		/* Determine alias port */
976 		if (GetNewPort(la, lnk, alias_port_param) != 0) {
977 			free(lnk);
978 			return (NULL);
979 		}
980 		/* Link-type dependent initialization */
981 		switch (link_type) {
982 			struct tcp_dat *aux_tcp;
983 
984 		case LINK_ICMP:
985 			la->icmpLinkCount++;
986 			break;
987 		case LINK_UDP:
988 			la->udpLinkCount++;
989 			break;
990 		case LINK_TCP:
991 			aux_tcp = malloc(sizeof(struct tcp_dat));
992 			if (aux_tcp != NULL) {
993 				int i;
994 
995 				la->tcpLinkCount++;
996 				aux_tcp->state.in = ALIAS_TCP_STATE_NOT_CONNECTED;
997 				aux_tcp->state.out = ALIAS_TCP_STATE_NOT_CONNECTED;
998 				aux_tcp->state.index = 0;
999 				aux_tcp->state.ack_modified = 0;
1000 				for (i = 0; i < N_LINK_TCP_DATA; i++)
1001 					aux_tcp->ack[i].active = 0;
1002 				aux_tcp->fwhole = -1;
1003 				lnk->data.tcp = aux_tcp;
1004 			} else {
1005 #ifdef LIBALIAS_DEBUG
1006 				fprintf(stderr, "PacketAlias/AddLink: ");
1007 				fprintf(stderr, " cannot allocate auxiliary TCP data\n");
1008 #endif
1009 				free(lnk);
1010 				return (NULL);
1011 			}
1012 			break;
1013 		case LINK_PPTP:
1014 			la->pptpLinkCount++;
1015 			break;
1016 		case LINK_FRAGMENT_ID:
1017 			la->fragmentIdLinkCount++;
1018 			break;
1019 		case LINK_FRAGMENT_PTR:
1020 			la->fragmentPtrLinkCount++;
1021 			break;
1022 		case LINK_ADDR:
1023 			break;
1024 		default:
1025 			la->protoLinkCount++;
1026 			break;
1027 		}
1028 
1029 		/* Set up pointers for output lookup table */
1030 		start_point = StartPointOut(src_addr, dst_addr,
1031 		    src_port, dst_port, link_type);
1032 		LIST_INSERT_HEAD(&la->linkTableOut[start_point], lnk, list_out);
1033 
1034 		/* Set up pointers for input lookup table */
1035 		start_point = StartPointIn(alias_addr, lnk->alias_port, link_type);
1036 		LIST_INSERT_HEAD(&la->linkTableIn[start_point], lnk, list_in);
1037 	} else {
1038 #ifdef LIBALIAS_DEBUG
1039 		fprintf(stderr, "PacketAlias/AddLink(): ");
1040 		fprintf(stderr, "malloc() call failed.\n");
1041 #endif
1042 	}
1043 	if (la->packetAliasMode & PKT_ALIAS_LOG) {
1044 		ShowAliasStats(la);
1045 	}
1046 	return (lnk);
1047 }
1048 
1049 static struct alias_link *
1050 ReLink(struct alias_link *old_lnk,
1051     struct in_addr src_addr,
1052     struct in_addr dst_addr,
1053     struct in_addr alias_addr,
1054     u_short src_port,
1055     u_short dst_port,
1056     int alias_port_param,	/* if less than zero, alias   */
1057     int link_type)
1058 {				/* port will be automatically *//* chosen.
1059 				 * If greater than    */
1060 	struct alias_link *new_lnk;	/* zero, equal to alias port  */
1061 	struct libalias *la = old_lnk->la;
1062 
1063 	LIBALIAS_LOCK_ASSERT(la);
1064 	new_lnk = AddLink(la, src_addr, dst_addr, alias_addr,
1065 	    src_port, dst_port, alias_port_param,
1066 	    link_type);
1067 #ifndef NO_FW_PUNCH
1068 	if (new_lnk != NULL &&
1069 	    old_lnk->link_type == LINK_TCP &&
1070 	    old_lnk->data.tcp->fwhole > 0) {
1071 		PunchFWHole(new_lnk);
1072 	}
1073 #endif
1074 	DeleteLink(old_lnk);
1075 	return (new_lnk);
1076 }
1077 
1078 static struct alias_link *
1079 _FindLinkOut(struct libalias *la, struct in_addr src_addr,
1080     struct in_addr dst_addr,
1081     u_short src_port,
1082     u_short dst_port,
1083     int link_type,
1084     int replace_partial_links)
1085 {
1086 	u_int i;
1087 	struct alias_link *lnk;
1088 
1089 	LIBALIAS_LOCK_ASSERT(la);
1090 	i = StartPointOut(src_addr, dst_addr, src_port, dst_port, link_type);
1091 	LIST_FOREACH(lnk, &la->linkTableOut[i], list_out) {
1092 		if (lnk->dst_addr.s_addr == dst_addr.s_addr &&
1093 		    lnk->src_addr.s_addr == src_addr.s_addr &&
1094 		    lnk->src_port == src_port &&
1095 		    lnk->dst_port == dst_port &&
1096 		    lnk->link_type == link_type &&
1097 		    lnk->server == NULL) {
1098 			lnk->timestamp = la->timeStamp;
1099 			break;
1100 		}
1101 	}
1102 
1103 /* Search for partially specified links. */
1104 	if (lnk == NULL && replace_partial_links) {
1105 		if (dst_port != 0 && dst_addr.s_addr != INADDR_ANY) {
1106 			lnk = _FindLinkOut(la, src_addr, dst_addr, src_port, 0,
1107 			    link_type, 0);
1108 			if (lnk == NULL)
1109 				lnk = _FindLinkOut(la, src_addr, la->nullAddress, src_port,
1110 				    dst_port, link_type, 0);
1111 		}
1112 		if (lnk == NULL &&
1113 		    (dst_port != 0 || dst_addr.s_addr != INADDR_ANY)) {
1114 			lnk = _FindLinkOut(la, src_addr, la->nullAddress, src_port, 0,
1115 			    link_type, 0);
1116 		}
1117 		if (lnk != NULL) {
1118 			lnk = ReLink(lnk,
1119 			    src_addr, dst_addr, lnk->alias_addr,
1120 			    src_port, dst_port, lnk->alias_port,
1121 			    link_type);
1122 		}
1123 	}
1124 	return (lnk);
1125 }
1126 
1127 static struct alias_link *
1128 FindLinkOut(struct libalias *la, struct in_addr src_addr,
1129     struct in_addr dst_addr,
1130     u_short src_port,
1131     u_short dst_port,
1132     int link_type,
1133     int replace_partial_links)
1134 {
1135 	struct alias_link *lnk;
1136 
1137 	LIBALIAS_LOCK_ASSERT(la);
1138 	lnk = _FindLinkOut(la, src_addr, dst_addr, src_port, dst_port,
1139 	    link_type, replace_partial_links);
1140 
1141 	if (lnk == NULL) {
1142 		/*
1143 		 * The following allows permanent links to be specified as
1144 		 * using the default source address (i.e. device interface
1145 		 * address) without knowing in advance what that address
1146 		 * is.
1147 		 */
1148 		if (la->aliasAddress.s_addr != INADDR_ANY &&
1149 		    src_addr.s_addr == la->aliasAddress.s_addr) {
1150 			lnk = _FindLinkOut(la, la->nullAddress, dst_addr, src_port, dst_port,
1151 			    link_type, replace_partial_links);
1152 		}
1153 	}
1154 	return (lnk);
1155 }
1156 
1157 static struct alias_link *
1158 _FindLinkIn(struct libalias *la, struct in_addr dst_addr,
1159     struct in_addr alias_addr,
1160     u_short dst_port,
1161     u_short alias_port,
1162     int link_type,
1163     int replace_partial_links)
1164 {
1165 	int flags_in;
1166 	u_int start_point;
1167 	struct alias_link *lnk;
1168 	struct alias_link *lnk_fully_specified;
1169 	struct alias_link *lnk_unknown_all;
1170 	struct alias_link *lnk_unknown_dst_addr;
1171 	struct alias_link *lnk_unknown_dst_port;
1172 
1173 	LIBALIAS_LOCK_ASSERT(la);
1174 /* Initialize pointers */
1175 	lnk_fully_specified = NULL;
1176 	lnk_unknown_all = NULL;
1177 	lnk_unknown_dst_addr = NULL;
1178 	lnk_unknown_dst_port = NULL;
1179 
1180 /* If either the dest addr or port is unknown, the search
1181    loop will have to know about this. */
1182 
1183 	flags_in = 0;
1184 	if (dst_addr.s_addr == INADDR_ANY)
1185 		flags_in |= LINK_UNKNOWN_DEST_ADDR;
1186 	if (dst_port == 0)
1187 		flags_in |= LINK_UNKNOWN_DEST_PORT;
1188 
1189 /* Search loop */
1190 	start_point = StartPointIn(alias_addr, alias_port, link_type);
1191 	LIST_FOREACH(lnk, &la->linkTableIn[start_point], list_in) {
1192 		int flags;
1193 
1194 		flags = flags_in | lnk->flags;
1195 		if (!(flags & LINK_PARTIALLY_SPECIFIED)) {
1196 			if (lnk->alias_addr.s_addr == alias_addr.s_addr
1197 			    && lnk->alias_port == alias_port
1198 			    && lnk->dst_addr.s_addr == dst_addr.s_addr
1199 			    && lnk->dst_port == dst_port
1200 			    && lnk->link_type == link_type) {
1201 				lnk_fully_specified = lnk;
1202 				break;
1203 			}
1204 		} else if ((flags & LINK_UNKNOWN_DEST_ADDR)
1205 		    && (flags & LINK_UNKNOWN_DEST_PORT)) {
1206 			if (lnk->alias_addr.s_addr == alias_addr.s_addr
1207 			    && lnk->alias_port == alias_port
1208 			    && lnk->link_type == link_type) {
1209 				if (lnk_unknown_all == NULL)
1210 					lnk_unknown_all = lnk;
1211 			}
1212 		} else if (flags & LINK_UNKNOWN_DEST_ADDR) {
1213 			if (lnk->alias_addr.s_addr == alias_addr.s_addr
1214 			    && lnk->alias_port == alias_port
1215 			    && lnk->link_type == link_type
1216 			    && lnk->dst_port == dst_port) {
1217 				if (lnk_unknown_dst_addr == NULL)
1218 					lnk_unknown_dst_addr = lnk;
1219 			}
1220 		} else if (flags & LINK_UNKNOWN_DEST_PORT) {
1221 			if (lnk->alias_addr.s_addr == alias_addr.s_addr
1222 			    && lnk->alias_port == alias_port
1223 			    && lnk->link_type == link_type
1224 			    && lnk->dst_addr.s_addr == dst_addr.s_addr) {
1225 				if (lnk_unknown_dst_port == NULL)
1226 					lnk_unknown_dst_port = lnk;
1227 			}
1228 		}
1229 	}
1230 
1231 	if (lnk_fully_specified != NULL) {
1232 		lnk_fully_specified->timestamp = la->timeStamp;
1233 		lnk = lnk_fully_specified;
1234 	} else if (lnk_unknown_dst_port != NULL)
1235 		lnk = lnk_unknown_dst_port;
1236 	else if (lnk_unknown_dst_addr != NULL)
1237 		lnk = lnk_unknown_dst_addr;
1238 	else if (lnk_unknown_all != NULL)
1239 		lnk = lnk_unknown_all;
1240 	else
1241 		return (NULL);
1242 
1243 	if (replace_partial_links &&
1244 	    (lnk->flags & LINK_PARTIALLY_SPECIFIED || lnk->server != NULL)) {
1245 		struct in_addr src_addr;
1246 		u_short src_port;
1247 
1248 		if (lnk->server != NULL) {	/* LSNAT link */
1249 			src_addr = lnk->server->addr;
1250 			src_port = lnk->server->port;
1251 			lnk->server = lnk->server->next;
1252 		} else {
1253 			src_addr = lnk->src_addr;
1254 			src_port = lnk->src_port;
1255 		}
1256 
1257 		if (link_type == LINK_SCTP) {
1258 		  lnk->src_addr = src_addr;
1259 		  lnk->src_port = src_port;
1260 		  return(lnk);
1261 		}
1262 		lnk = ReLink(lnk,
1263 		    src_addr, dst_addr, alias_addr,
1264 		    src_port, dst_port, alias_port,
1265 		    link_type);
1266 	}
1267 	return (lnk);
1268 }
1269 
1270 static struct alias_link *
1271 FindLinkIn(struct libalias *la, struct in_addr dst_addr,
1272     struct in_addr alias_addr,
1273     u_short dst_port,
1274     u_short alias_port,
1275     int link_type,
1276     int replace_partial_links)
1277 {
1278 	struct alias_link *lnk;
1279 
1280 	LIBALIAS_LOCK_ASSERT(la);
1281 	lnk = _FindLinkIn(la, dst_addr, alias_addr, dst_port, alias_port,
1282 	    link_type, replace_partial_links);
1283 
1284 	if (lnk == NULL) {
1285 		/*
1286 		 * The following allows permanent links to be specified as
1287 		 * using the default aliasing address (i.e. device
1288 		 * interface address) without knowing in advance what that
1289 		 * address is.
1290 		 */
1291 		if (la->aliasAddress.s_addr != INADDR_ANY &&
1292 		    alias_addr.s_addr == la->aliasAddress.s_addr) {
1293 			lnk = _FindLinkIn(la, dst_addr, la->nullAddress, dst_port, alias_port,
1294 			    link_type, replace_partial_links);
1295 		}
1296 	}
1297 	return (lnk);
1298 }
1299 
1300 /* External routines for finding/adding links
1301 
1302 -- "external" means outside alias_db.c, but within alias*.c --
1303 
1304     FindIcmpIn(), FindIcmpOut()
1305     FindFragmentIn1(), FindFragmentIn2()
1306     AddFragmentPtrLink(), FindFragmentPtr()
1307     FindProtoIn(), FindProtoOut()
1308     FindUdpTcpIn(), FindUdpTcpOut()
1309     AddPptp(), FindPptpOutByCallId(), FindPptpInByCallId(),
1310     FindPptpOutByPeerCallId(), FindPptpInByPeerCallId()
1311     FindOriginalAddress(), FindAliasAddress()
1312 
1313 (prototypes in alias_local.h)
1314 */
1315 
1316 struct alias_link *
1317 FindIcmpIn(struct libalias *la, struct in_addr dst_addr,
1318     struct in_addr alias_addr,
1319     u_short id_alias,
1320     int create)
1321 {
1322 	struct alias_link *lnk;
1323 
1324 	LIBALIAS_LOCK_ASSERT(la);
1325 	lnk = FindLinkIn(la, dst_addr, alias_addr,
1326 	    NO_DEST_PORT, id_alias,
1327 	    LINK_ICMP, 0);
1328 	if (lnk == NULL && create && !(la->packetAliasMode & PKT_ALIAS_DENY_INCOMING)) {
1329 		struct in_addr target_addr;
1330 
1331 		target_addr = FindOriginalAddress(la, alias_addr);
1332 		lnk = AddLink(la, target_addr, dst_addr, alias_addr,
1333 		    id_alias, NO_DEST_PORT, id_alias,
1334 		    LINK_ICMP);
1335 	}
1336 	return (lnk);
1337 }
1338 
1339 struct alias_link *
1340 FindIcmpOut(struct libalias *la, struct in_addr src_addr,
1341     struct in_addr dst_addr,
1342     u_short id,
1343     int create)
1344 {
1345 	struct alias_link *lnk;
1346 
1347 	LIBALIAS_LOCK_ASSERT(la);
1348 	lnk = FindLinkOut(la, src_addr, dst_addr,
1349 	    id, NO_DEST_PORT,
1350 	    LINK_ICMP, 0);
1351 	if (lnk == NULL && create) {
1352 		struct in_addr alias_addr;
1353 
1354 		alias_addr = FindAliasAddress(la, src_addr);
1355 		lnk = AddLink(la, src_addr, dst_addr, alias_addr,
1356 		    id, NO_DEST_PORT, GET_ALIAS_ID,
1357 		    LINK_ICMP);
1358 	}
1359 	return (lnk);
1360 }
1361 
1362 struct alias_link *
1363 FindFragmentIn1(struct libalias *la, struct in_addr dst_addr,
1364     struct in_addr alias_addr,
1365     u_short ip_id)
1366 {
1367 	struct alias_link *lnk;
1368 
1369 	LIBALIAS_LOCK_ASSERT(la);
1370 	lnk = FindLinkIn(la, dst_addr, alias_addr,
1371 	    NO_DEST_PORT, ip_id,
1372 	    LINK_FRAGMENT_ID, 0);
1373 
1374 	if (lnk == NULL) {
1375 		lnk = AddLink(la, la->nullAddress, dst_addr, alias_addr,
1376 		    NO_SRC_PORT, NO_DEST_PORT, ip_id,
1377 		    LINK_FRAGMENT_ID);
1378 	}
1379 	return (lnk);
1380 }
1381 
1382 struct alias_link *
1383 FindFragmentIn2(struct libalias *la, struct in_addr dst_addr,	/* Doesn't add a link if
1384 								 * one */
1385     struct in_addr alias_addr,	/* is not found.           */
1386     u_short ip_id)
1387 {
1388 
1389 	LIBALIAS_LOCK_ASSERT(la);
1390 	return FindLinkIn(la, dst_addr, alias_addr,
1391 	    NO_DEST_PORT, ip_id,
1392 	    LINK_FRAGMENT_ID, 0);
1393 }
1394 
1395 struct alias_link *
1396 AddFragmentPtrLink(struct libalias *la, struct in_addr dst_addr,
1397     u_short ip_id)
1398 {
1399 
1400 	LIBALIAS_LOCK_ASSERT(la);
1401 	return AddLink(la, la->nullAddress, dst_addr, la->nullAddress,
1402 	    NO_SRC_PORT, NO_DEST_PORT, ip_id,
1403 	    LINK_FRAGMENT_PTR);
1404 }
1405 
1406 struct alias_link *
1407 FindFragmentPtr(struct libalias *la, struct in_addr dst_addr,
1408     u_short ip_id)
1409 {
1410 
1411 	LIBALIAS_LOCK_ASSERT(la);
1412 	return FindLinkIn(la, dst_addr, la->nullAddress,
1413 	    NO_DEST_PORT, ip_id,
1414 	    LINK_FRAGMENT_PTR, 0);
1415 }
1416 
1417 struct alias_link *
1418 FindProtoIn(struct libalias *la, struct in_addr dst_addr,
1419     struct in_addr alias_addr,
1420     u_char proto)
1421 {
1422 	struct alias_link *lnk;
1423 
1424 	LIBALIAS_LOCK_ASSERT(la);
1425 	lnk = FindLinkIn(la, dst_addr, alias_addr,
1426 	    NO_DEST_PORT, 0,
1427 	    proto, 1);
1428 
1429 	if (lnk == NULL && !(la->packetAliasMode & PKT_ALIAS_DENY_INCOMING)) {
1430 		struct in_addr target_addr;
1431 
1432 		target_addr = FindOriginalAddress(la, alias_addr);
1433 		lnk = AddLink(la, target_addr, dst_addr, alias_addr,
1434 		    NO_SRC_PORT, NO_DEST_PORT, 0,
1435 		    proto);
1436 	}
1437 	return (lnk);
1438 }
1439 
1440 struct alias_link *
1441 FindProtoOut(struct libalias *la, struct in_addr src_addr,
1442     struct in_addr dst_addr,
1443     u_char proto)
1444 {
1445 	struct alias_link *lnk;
1446 
1447 	LIBALIAS_LOCK_ASSERT(la);
1448 	lnk = FindLinkOut(la, src_addr, dst_addr,
1449 	    NO_SRC_PORT, NO_DEST_PORT,
1450 	    proto, 1);
1451 
1452 	if (lnk == NULL) {
1453 		struct in_addr alias_addr;
1454 
1455 		alias_addr = FindAliasAddress(la, src_addr);
1456 		lnk = AddLink(la, src_addr, dst_addr, alias_addr,
1457 		    NO_SRC_PORT, NO_DEST_PORT, 0,
1458 		    proto);
1459 	}
1460 	return (lnk);
1461 }
1462 
1463 struct alias_link *
1464 FindUdpTcpIn(struct libalias *la, struct in_addr dst_addr,
1465     struct in_addr alias_addr,
1466     u_short dst_port,
1467     u_short alias_port,
1468     u_char proto,
1469     int create)
1470 {
1471 	int link_type;
1472 	struct alias_link *lnk;
1473 
1474 	LIBALIAS_LOCK_ASSERT(la);
1475 	switch (proto) {
1476 	case IPPROTO_UDP:
1477 		link_type = LINK_UDP;
1478 		break;
1479 	case IPPROTO_TCP:
1480 		link_type = LINK_TCP;
1481 		break;
1482 	default:
1483 		return (NULL);
1484 		break;
1485 	}
1486 
1487 	lnk = FindLinkIn(la, dst_addr, alias_addr,
1488 	    dst_port, alias_port,
1489 	    link_type, create);
1490 
1491 	if (lnk == NULL && create && !(la->packetAliasMode & PKT_ALIAS_DENY_INCOMING)) {
1492 		struct in_addr target_addr;
1493 
1494 		target_addr = FindOriginalAddress(la, alias_addr);
1495 		lnk = AddLink(la, target_addr, dst_addr, alias_addr,
1496 		    alias_port, dst_port, alias_port,
1497 		    link_type);
1498 	}
1499 	return (lnk);
1500 }
1501 
1502 struct alias_link *
1503 FindUdpTcpOut(struct libalias *la, struct in_addr src_addr,
1504     struct in_addr dst_addr,
1505     u_short src_port,
1506     u_short dst_port,
1507     u_char proto,
1508     int create)
1509 {
1510 	int link_type;
1511 	struct alias_link *lnk;
1512 
1513 	LIBALIAS_LOCK_ASSERT(la);
1514 	switch (proto) {
1515 	case IPPROTO_UDP:
1516 		link_type = LINK_UDP;
1517 		break;
1518 	case IPPROTO_TCP:
1519 		link_type = LINK_TCP;
1520 		break;
1521 	default:
1522 		return (NULL);
1523 		break;
1524 	}
1525 
1526 	lnk = FindLinkOut(la, src_addr, dst_addr, src_port, dst_port, link_type, create);
1527 
1528 	if (lnk == NULL && create) {
1529 		struct in_addr alias_addr;
1530 
1531 		alias_addr = FindAliasAddress(la, src_addr);
1532 		lnk = AddLink(la, src_addr, dst_addr, alias_addr,
1533 		    src_port, dst_port, GET_ALIAS_PORT,
1534 		    link_type);
1535 	}
1536 	return (lnk);
1537 }
1538 
1539 struct alias_link *
1540 AddPptp(struct libalias *la, struct in_addr src_addr,
1541     struct in_addr dst_addr,
1542     struct in_addr alias_addr,
1543     u_int16_t src_call_id)
1544 {
1545 	struct alias_link *lnk;
1546 
1547 	LIBALIAS_LOCK_ASSERT(la);
1548 	lnk = AddLink(la, src_addr, dst_addr, alias_addr,
1549 	    src_call_id, 0, GET_ALIAS_PORT,
1550 	    LINK_PPTP);
1551 
1552 	return (lnk);
1553 }
1554 
1555 struct alias_link *
1556 FindPptpOutByCallId(struct libalias *la, struct in_addr src_addr,
1557     struct in_addr dst_addr,
1558     u_int16_t src_call_id)
1559 {
1560 	u_int i;
1561 	struct alias_link *lnk;
1562 
1563 	LIBALIAS_LOCK_ASSERT(la);
1564 	i = StartPointOut(src_addr, dst_addr, 0, 0, LINK_PPTP);
1565 	LIST_FOREACH(lnk, &la->linkTableOut[i], list_out)
1566 	    if (lnk->link_type == LINK_PPTP &&
1567 	    lnk->src_addr.s_addr == src_addr.s_addr &&
1568 	    lnk->dst_addr.s_addr == dst_addr.s_addr &&
1569 	    lnk->src_port == src_call_id)
1570 		break;
1571 
1572 	return (lnk);
1573 }
1574 
1575 struct alias_link *
1576 FindPptpOutByPeerCallId(struct libalias *la, struct in_addr src_addr,
1577     struct in_addr dst_addr,
1578     u_int16_t dst_call_id)
1579 {
1580 	u_int i;
1581 	struct alias_link *lnk;
1582 
1583 	LIBALIAS_LOCK_ASSERT(la);
1584 	i = StartPointOut(src_addr, dst_addr, 0, 0, LINK_PPTP);
1585 	LIST_FOREACH(lnk, &la->linkTableOut[i], list_out)
1586 	    if (lnk->link_type == LINK_PPTP &&
1587 	    lnk->src_addr.s_addr == src_addr.s_addr &&
1588 	    lnk->dst_addr.s_addr == dst_addr.s_addr &&
1589 	    lnk->dst_port == dst_call_id)
1590 		break;
1591 
1592 	return (lnk);
1593 }
1594 
1595 struct alias_link *
1596 FindPptpInByCallId(struct libalias *la, struct in_addr dst_addr,
1597     struct in_addr alias_addr,
1598     u_int16_t dst_call_id)
1599 {
1600 	u_int i;
1601 	struct alias_link *lnk;
1602 
1603 	LIBALIAS_LOCK_ASSERT(la);
1604 	i = StartPointIn(alias_addr, 0, LINK_PPTP);
1605 	LIST_FOREACH(lnk, &la->linkTableIn[i], list_in)
1606 	    if (lnk->link_type == LINK_PPTP &&
1607 	    lnk->dst_addr.s_addr == dst_addr.s_addr &&
1608 	    lnk->alias_addr.s_addr == alias_addr.s_addr &&
1609 	    lnk->dst_port == dst_call_id)
1610 		break;
1611 
1612 	return (lnk);
1613 }
1614 
1615 struct alias_link *
1616 FindPptpInByPeerCallId(struct libalias *la, struct in_addr dst_addr,
1617     struct in_addr alias_addr,
1618     u_int16_t alias_call_id)
1619 {
1620 	struct alias_link *lnk;
1621 
1622 	LIBALIAS_LOCK_ASSERT(la);
1623 	lnk = FindLinkIn(la, dst_addr, alias_addr,
1624 	    0 /* any */ , alias_call_id,
1625 	    LINK_PPTP, 0);
1626 
1627 	return (lnk);
1628 }
1629 
1630 struct alias_link *
1631 FindRtspOut(struct libalias *la, struct in_addr src_addr,
1632     struct in_addr dst_addr,
1633     u_short src_port,
1634     u_short alias_port,
1635     u_char proto)
1636 {
1637 	int link_type;
1638 	struct alias_link *lnk;
1639 
1640 	LIBALIAS_LOCK_ASSERT(la);
1641 	switch (proto) {
1642 	case IPPROTO_UDP:
1643 		link_type = LINK_UDP;
1644 		break;
1645 	case IPPROTO_TCP:
1646 		link_type = LINK_TCP;
1647 		break;
1648 	default:
1649 		return (NULL);
1650 		break;
1651 	}
1652 
1653 	lnk = FindLinkOut(la, src_addr, dst_addr, src_port, 0, link_type, 1);
1654 
1655 	if (lnk == NULL) {
1656 		struct in_addr alias_addr;
1657 
1658 		alias_addr = FindAliasAddress(la, src_addr);
1659 		lnk = AddLink(la, src_addr, dst_addr, alias_addr,
1660 		    src_port, 0, alias_port,
1661 		    link_type);
1662 	}
1663 	return (lnk);
1664 }
1665 
1666 struct in_addr
1667 FindOriginalAddress(struct libalias *la, struct in_addr alias_addr)
1668 {
1669 	struct alias_link *lnk;
1670 
1671 	LIBALIAS_LOCK_ASSERT(la);
1672 	lnk = FindLinkIn(la, la->nullAddress, alias_addr,
1673 	    0, 0, LINK_ADDR, 0);
1674 	if (lnk == NULL) {
1675 		la->newDefaultLink = 1;
1676 		if (la->targetAddress.s_addr == INADDR_ANY)
1677 			return (alias_addr);
1678 		else if (la->targetAddress.s_addr == INADDR_NONE)
1679 			return (la->aliasAddress.s_addr != INADDR_ANY) ?
1680 			    la->aliasAddress : alias_addr;
1681 		else
1682 			return (la->targetAddress);
1683 	} else {
1684 		if (lnk->server != NULL) {	/* LSNAT link */
1685 			struct in_addr src_addr;
1686 
1687 			src_addr = lnk->server->addr;
1688 			lnk->server = lnk->server->next;
1689 			return (src_addr);
1690 		} else if (lnk->src_addr.s_addr == INADDR_ANY)
1691 			return (la->aliasAddress.s_addr != INADDR_ANY) ?
1692 			    la->aliasAddress : alias_addr;
1693 		else
1694 			return (lnk->src_addr);
1695 	}
1696 }
1697 
1698 struct in_addr
1699 FindAliasAddress(struct libalias *la, struct in_addr original_addr)
1700 {
1701 	struct alias_link *lnk;
1702 
1703 	LIBALIAS_LOCK_ASSERT(la);
1704 	lnk = FindLinkOut(la, original_addr, la->nullAddress,
1705 	    0, 0, LINK_ADDR, 0);
1706 	if (lnk == NULL) {
1707 		return (la->aliasAddress.s_addr != INADDR_ANY) ?
1708 		    la->aliasAddress : original_addr;
1709 	} else {
1710 		if (lnk->alias_addr.s_addr == INADDR_ANY)
1711 			return (la->aliasAddress.s_addr != INADDR_ANY) ?
1712 			    la->aliasAddress : original_addr;
1713 		else
1714 			return (lnk->alias_addr);
1715 	}
1716 }
1717 
1718 /* External routines for getting or changing link data
1719    (external to alias_db.c, but internal to alias*.c)
1720 
1721     SetFragmentData(), GetFragmentData()
1722     SetFragmentPtr(), GetFragmentPtr()
1723     SetStateIn(), SetStateOut(), GetStateIn(), GetStateOut()
1724     GetOriginalAddress(), GetDestAddress(), GetAliasAddress()
1725     GetOriginalPort(), GetAliasPort()
1726     SetAckModified(), GetAckModified()
1727     GetDeltaAckIn(), GetDeltaSeqOut(), AddSeq()
1728     SetProtocolFlags(), GetProtocolFlags()
1729     SetDestCallId()
1730 */
1731 
1732 void
1733 SetFragmentAddr(struct alias_link *lnk, struct in_addr src_addr)
1734 {
1735 	lnk->data.frag_addr = src_addr;
1736 }
1737 
1738 void
1739 GetFragmentAddr(struct alias_link *lnk, struct in_addr *src_addr)
1740 {
1741 	*src_addr = lnk->data.frag_addr;
1742 }
1743 
1744 void
1745 SetFragmentPtr(struct alias_link *lnk, void *fptr)
1746 {
1747 	lnk->data.frag_ptr = fptr;
1748 }
1749 
1750 void
1751 GetFragmentPtr(struct alias_link *lnk, void **fptr)
1752 {
1753 	*fptr = lnk->data.frag_ptr;
1754 }
1755 
1756 void
1757 SetStateIn(struct alias_link *lnk, int state)
1758 {
1759 	/* TCP input state */
1760 	switch (state) {
1761 		case ALIAS_TCP_STATE_DISCONNECTED:
1762 		if (lnk->data.tcp->state.out != ALIAS_TCP_STATE_CONNECTED)
1763 			lnk->expire_time = TCP_EXPIRE_DEAD;
1764 		else
1765 			lnk->expire_time = TCP_EXPIRE_SINGLEDEAD;
1766 		break;
1767 	case ALIAS_TCP_STATE_CONNECTED:
1768 		if (lnk->data.tcp->state.out == ALIAS_TCP_STATE_CONNECTED)
1769 			lnk->expire_time = TCP_EXPIRE_CONNECTED;
1770 		break;
1771 	default:
1772 #ifdef	_KERNEL
1773 		panic("libalias:SetStateIn() unknown state");
1774 #else
1775 		abort();
1776 #endif
1777 	}
1778 	lnk->data.tcp->state.in = state;
1779 }
1780 
1781 void
1782 SetStateOut(struct alias_link *lnk, int state)
1783 {
1784 	/* TCP output state */
1785 	switch (state) {
1786 		case ALIAS_TCP_STATE_DISCONNECTED:
1787 		if (lnk->data.tcp->state.in != ALIAS_TCP_STATE_CONNECTED)
1788 			lnk->expire_time = TCP_EXPIRE_DEAD;
1789 		else
1790 			lnk->expire_time = TCP_EXPIRE_SINGLEDEAD;
1791 		break;
1792 	case ALIAS_TCP_STATE_CONNECTED:
1793 		if (lnk->data.tcp->state.in == ALIAS_TCP_STATE_CONNECTED)
1794 			lnk->expire_time = TCP_EXPIRE_CONNECTED;
1795 		break;
1796 	default:
1797 #ifdef	_KERNEL
1798 		panic("libalias:SetStateOut() unknown state");
1799 #else
1800 		abort();
1801 #endif
1802 	}
1803 	lnk->data.tcp->state.out = state;
1804 }
1805 
1806 int
1807 GetStateIn(struct alias_link *lnk)
1808 {
1809 	/* TCP input state */
1810 	return (lnk->data.tcp->state.in);
1811 }
1812 
1813 int
1814 GetStateOut(struct alias_link *lnk)
1815 {
1816 	/* TCP output state */
1817 	return (lnk->data.tcp->state.out);
1818 }
1819 
1820 struct in_addr
1821 GetOriginalAddress(struct alias_link *lnk)
1822 {
1823 	if (lnk->src_addr.s_addr == INADDR_ANY)
1824 		return (lnk->la->aliasAddress);
1825 	else
1826 		return (lnk->src_addr);
1827 }
1828 
1829 struct in_addr
1830 GetDestAddress(struct alias_link *lnk)
1831 {
1832 	return (lnk->dst_addr);
1833 }
1834 
1835 struct in_addr
1836 GetAliasAddress(struct alias_link *lnk)
1837 {
1838 	if (lnk->alias_addr.s_addr == INADDR_ANY)
1839 		return (lnk->la->aliasAddress);
1840 	else
1841 		return (lnk->alias_addr);
1842 }
1843 
1844 struct in_addr
1845 GetDefaultAliasAddress(struct libalias *la)
1846 {
1847 
1848 	LIBALIAS_LOCK_ASSERT(la);
1849 	return (la->aliasAddress);
1850 }
1851 
1852 void
1853 SetDefaultAliasAddress(struct libalias *la, struct in_addr alias_addr)
1854 {
1855 
1856 	LIBALIAS_LOCK_ASSERT(la);
1857 	la->aliasAddress = alias_addr;
1858 }
1859 
1860 u_short
1861 GetOriginalPort(struct alias_link *lnk)
1862 {
1863 	return (lnk->src_port);
1864 }
1865 
1866 u_short
1867 GetAliasPort(struct alias_link *lnk)
1868 {
1869 	return (lnk->alias_port);
1870 }
1871 
1872 #ifndef NO_FW_PUNCH
1873 static		u_short
1874 GetDestPort(struct alias_link *lnk)
1875 {
1876 	return (lnk->dst_port);
1877 }
1878 
1879 #endif
1880 
1881 void
1882 SetAckModified(struct alias_link *lnk)
1883 {
1884 /* Indicate that ACK numbers have been modified in a TCP connection */
1885 	lnk->data.tcp->state.ack_modified = 1;
1886 }
1887 
1888 struct in_addr
1889 GetProxyAddress(struct alias_link *lnk)
1890 {
1891 	return (lnk->proxy_addr);
1892 }
1893 
1894 void
1895 SetProxyAddress(struct alias_link *lnk, struct in_addr addr)
1896 {
1897 	lnk->proxy_addr = addr;
1898 }
1899 
1900 u_short
1901 GetProxyPort(struct alias_link *lnk)
1902 {
1903 	return (lnk->proxy_port);
1904 }
1905 
1906 void
1907 SetProxyPort(struct alias_link *lnk, u_short port)
1908 {
1909 	lnk->proxy_port = port;
1910 }
1911 
1912 int
1913 GetAckModified(struct alias_link *lnk)
1914 {
1915 /* See if ACK numbers have been modified */
1916 	return (lnk->data.tcp->state.ack_modified);
1917 }
1918 
1919 // XXX ip free
1920 int
1921 GetDeltaAckIn(u_long ack, struct alias_link *lnk)
1922 {
1923 /*
1924 Find out how much the ACK number has been altered for an incoming
1925 TCP packet.  To do this, a circular list of ACK numbers where the TCP
1926 packet size was altered is searched.
1927 */
1928 
1929 	int i;
1930 	int delta, ack_diff_min;
1931 
1932 	delta = 0;
1933 	ack_diff_min = -1;
1934 	for (i = 0; i < N_LINK_TCP_DATA; i++) {
1935 		struct ack_data_record x;
1936 
1937 		x = lnk->data.tcp->ack[i];
1938 		if (x.active == 1) {
1939 			int ack_diff;
1940 
1941 			ack_diff = SeqDiff(x.ack_new, ack);
1942 			if (ack_diff >= 0) {
1943 				if (ack_diff_min >= 0) {
1944 					if (ack_diff < ack_diff_min) {
1945 						delta = x.delta;
1946 						ack_diff_min = ack_diff;
1947 					}
1948 				} else {
1949 					delta = x.delta;
1950 					ack_diff_min = ack_diff;
1951 				}
1952 			}
1953 		}
1954 	}
1955 	return (delta);
1956 }
1957 
1958 // XXX ip free
1959 int
1960 GetDeltaSeqOut(u_long seq, struct alias_link *lnk)
1961 {
1962 /*
1963 Find out how much the sequence number has been altered for an outgoing
1964 TCP packet.  To do this, a circular list of ACK numbers where the TCP
1965 packet size was altered is searched.
1966 */
1967 
1968 	int i;
1969 	int delta, seq_diff_min;
1970 
1971 	delta = 0;
1972 	seq_diff_min = -1;
1973 	for (i = 0; i < N_LINK_TCP_DATA; i++) {
1974 		struct ack_data_record x;
1975 
1976 		x = lnk->data.tcp->ack[i];
1977 		if (x.active == 1) {
1978 			int seq_diff;
1979 
1980 			seq_diff = SeqDiff(x.ack_old, seq);
1981 			if (seq_diff >= 0) {
1982 				if (seq_diff_min >= 0) {
1983 					if (seq_diff < seq_diff_min) {
1984 						delta = x.delta;
1985 						seq_diff_min = seq_diff;
1986 					}
1987 				} else {
1988 					delta = x.delta;
1989 					seq_diff_min = seq_diff;
1990 				}
1991 			}
1992 		}
1993 	}
1994 	return (delta);
1995 }
1996 
1997 // XXX ip free
1998 void
1999 AddSeq(struct alias_link *lnk, int delta, u_int ip_hl, u_short ip_len,
2000     u_long th_seq, u_int th_off)
2001 {
2002 /*
2003 When a TCP packet has been altered in length, save this
2004 information in a circular list.  If enough packets have
2005 been altered, then this list will begin to overwrite itself.
2006 */
2007 
2008 	struct ack_data_record x;
2009 	int hlen, tlen, dlen;
2010 	int i;
2011 
2012 	hlen = (ip_hl + th_off) << 2;
2013 	tlen = ntohs(ip_len);
2014 	dlen = tlen - hlen;
2015 
2016 	x.ack_old = htonl(ntohl(th_seq) + dlen);
2017 	x.ack_new = htonl(ntohl(th_seq) + dlen + delta);
2018 	x.delta = delta;
2019 	x.active = 1;
2020 
2021 	i = lnk->data.tcp->state.index;
2022 	lnk->data.tcp->ack[i] = x;
2023 
2024 	i++;
2025 	if (i == N_LINK_TCP_DATA)
2026 		lnk->data.tcp->state.index = 0;
2027 	else
2028 		lnk->data.tcp->state.index = i;
2029 }
2030 
2031 void
2032 SetExpire(struct alias_link *lnk, int expire)
2033 {
2034 	if (expire == 0) {
2035 		lnk->flags &= ~LINK_PERMANENT;
2036 		DeleteLink(lnk);
2037 	} else if (expire == -1) {
2038 		lnk->flags |= LINK_PERMANENT;
2039 	} else if (expire > 0) {
2040 		lnk->expire_time = expire;
2041 	} else {
2042 #ifdef LIBALIAS_DEBUG
2043 		fprintf(stderr, "PacketAlias/SetExpire(): ");
2044 		fprintf(stderr, "error in expire parameter\n");
2045 #endif
2046 	}
2047 }
2048 
2049 void
2050 ClearCheckNewLink(struct libalias *la)
2051 {
2052 
2053 	LIBALIAS_LOCK_ASSERT(la);
2054 	la->newDefaultLink = 0;
2055 }
2056 
2057 void
2058 SetProtocolFlags(struct alias_link *lnk, int pflags)
2059 {
2060 
2061 	lnk->pflags = pflags;
2062 }
2063 
2064 int
2065 GetProtocolFlags(struct alias_link *lnk)
2066 {
2067 
2068 	return (lnk->pflags);
2069 }
2070 
2071 void
2072 SetDestCallId(struct alias_link *lnk, u_int16_t cid)
2073 {
2074 	struct libalias *la = lnk->la;
2075 
2076 	LIBALIAS_LOCK_ASSERT(la);
2077 	la->deleteAllLinks = 1;
2078 	ReLink(lnk, lnk->src_addr, lnk->dst_addr, lnk->alias_addr,
2079 	    lnk->src_port, cid, lnk->alias_port, lnk->link_type);
2080 	la->deleteAllLinks = 0;
2081 }
2082 
2083 /* Miscellaneous Functions
2084 
2085     HouseKeeping()
2086     InitPacketAliasLog()
2087     UninitPacketAliasLog()
2088 */
2089 
2090 /*
2091     Whenever an outgoing or incoming packet is handled, HouseKeeping()
2092     is called to find and remove timed-out aliasing links.  Logic exists
2093     to sweep through the entire table and linked list structure
2094     every 60 seconds.
2095 
2096     (prototype in alias_local.h)
2097 */
2098 
2099 void
2100 HouseKeeping(struct libalias *la)
2101 {
2102 	int i, n;
2103 #ifndef	_KERNEL
2104 	struct timeval tv;
2105 #endif
2106 
2107 	LIBALIAS_LOCK_ASSERT(la);
2108 	/*
2109 	 * Save system time (seconds) in global variable timeStamp for use
2110 	 * by other functions. This is done so as not to unnecessarily
2111 	 * waste timeline by making system calls.
2112 	 */
2113 #ifdef	_KERNEL
2114 	la->timeStamp = time_uptime;
2115 #else
2116 	gettimeofday(&tv, NULL);
2117 	la->timeStamp = tv.tv_sec;
2118 #endif
2119 
2120 	/* Compute number of spokes (output table link chains) to cover */
2121 	n = LINK_TABLE_OUT_SIZE * (la->timeStamp - la->lastCleanupTime);
2122 	n /= ALIAS_CLEANUP_INTERVAL_SECS;
2123 
2124 	/* Handle different cases */
2125 	if (n > 0) {
2126 		if (n > ALIAS_CLEANUP_MAX_SPOKES)
2127 			n = ALIAS_CLEANUP_MAX_SPOKES;
2128 		la->lastCleanupTime = la->timeStamp;
2129 		for (i = 0; i < n; i++)
2130 			IncrementalCleanup(la);
2131 	} else if (n < 0) {
2132 #ifdef LIBALIAS_DEBUG
2133 		fprintf(stderr, "PacketAlias/HouseKeeping(): ");
2134 		fprintf(stderr, "something unexpected in time values\n");
2135 #endif
2136 		la->lastCleanupTime = la->timeStamp;
2137 	}
2138 }
2139 
2140 /* Init the log file and enable logging */
2141 static int
2142 InitPacketAliasLog(struct libalias *la)
2143 {
2144 
2145 	LIBALIAS_LOCK_ASSERT(la);
2146 	if (~la->packetAliasMode & PKT_ALIAS_LOG) {
2147 #ifdef _KERNEL
2148 		if ((la->logDesc = malloc(LIBALIAS_BUF_SIZE)))
2149 			;
2150 #else
2151 		if ((la->logDesc = fopen("/var/log/alias.log", "w")))
2152 			fprintf(la->logDesc, "PacketAlias/InitPacketAliasLog: Packet alias logging enabled.\n");
2153 #endif
2154 		else
2155 			return (ENOMEM); /* log initialization failed */
2156 		la->packetAliasMode |= PKT_ALIAS_LOG;
2157 	}
2158 
2159 	return (1);
2160 }
2161 
2162 /* Close the log-file and disable logging. */
2163 static void
2164 UninitPacketAliasLog(struct libalias *la)
2165 {
2166 
2167 	LIBALIAS_LOCK_ASSERT(la);
2168 	if (la->logDesc) {
2169 #ifdef _KERNEL
2170 		free(la->logDesc);
2171 #else
2172 		fclose(la->logDesc);
2173 #endif
2174 		la->logDesc = NULL;
2175 	}
2176 	la->packetAliasMode &= ~PKT_ALIAS_LOG;
2177 }
2178 
2179 /* Outside world interfaces
2180 
2181 -- "outside world" means other than alias*.c routines --
2182 
2183     PacketAliasRedirectPort()
2184     PacketAliasAddServer()
2185     PacketAliasRedirectProto()
2186     PacketAliasRedirectAddr()
2187     PacketAliasRedirectDynamic()
2188     PacketAliasRedirectDelete()
2189     PacketAliasSetAddress()
2190     PacketAliasInit()
2191     PacketAliasUninit()
2192     PacketAliasSetMode()
2193 
2194 (prototypes in alias.h)
2195 */
2196 
2197 /* Redirection from a specific public addr:port to a
2198    private addr:port */
2199 struct alias_link *
2200 LibAliasRedirectPort(struct libalias *la, struct in_addr src_addr, u_short src_port,
2201     struct in_addr dst_addr, u_short dst_port,
2202     struct in_addr alias_addr, u_short alias_port,
2203     u_char proto)
2204 {
2205 	int link_type;
2206 	struct alias_link *lnk;
2207 
2208 	LIBALIAS_LOCK(la);
2209 	switch (proto) {
2210 	case IPPROTO_UDP:
2211 		link_type = LINK_UDP;
2212 		break;
2213 	case IPPROTO_TCP:
2214 		link_type = LINK_TCP;
2215 		break;
2216 	case IPPROTO_SCTP:
2217 		link_type = LINK_SCTP;
2218 		break;
2219 	default:
2220 #ifdef LIBALIAS_DEBUG
2221 		fprintf(stderr, "PacketAliasRedirectPort(): ");
2222 		fprintf(stderr, "only SCTP, TCP and UDP protocols allowed\n");
2223 #endif
2224 		lnk = NULL;
2225 		goto getout;
2226 	}
2227 
2228 	lnk = AddLink(la, src_addr, dst_addr, alias_addr,
2229 	    src_port, dst_port, alias_port,
2230 	    link_type);
2231 
2232 	if (lnk != NULL) {
2233 		lnk->flags |= LINK_PERMANENT;
2234 	}
2235 #ifdef LIBALIAS_DEBUG
2236 	else {
2237 		fprintf(stderr, "PacketAliasRedirectPort(): "
2238 		    "call to AddLink() failed\n");
2239 	}
2240 #endif
2241 
2242 getout:
2243 	LIBALIAS_UNLOCK(la);
2244 	return (lnk);
2245 }
2246 
2247 /* Add server to the pool of servers */
2248 int
2249 LibAliasAddServer(struct libalias *la, struct alias_link *lnk, struct in_addr addr, u_short port)
2250 {
2251 	struct server *server;
2252 	int res;
2253 
2254 	LIBALIAS_LOCK(la);
2255 	(void)la;
2256 
2257 	server = malloc(sizeof(struct server));
2258 
2259 	if (server != NULL) {
2260 		struct server *head;
2261 
2262 		server->addr = addr;
2263 		server->port = port;
2264 
2265 		head = lnk->server;
2266 		if (head == NULL)
2267 			server->next = server;
2268 		else {
2269 			struct server *s;
2270 
2271 			for (s = head; s->next != head; s = s->next);
2272 			s->next = server;
2273 			server->next = head;
2274 		}
2275 		lnk->server = server;
2276 		res = 0;
2277 	} else
2278 		res = -1;
2279 
2280 	LIBALIAS_UNLOCK(la);
2281 	return (res);
2282 }
2283 
2284 /* Redirect packets of a given IP protocol from a specific
2285    public address to a private address */
2286 struct alias_link *
2287 LibAliasRedirectProto(struct libalias *la, struct in_addr src_addr,
2288     struct in_addr dst_addr,
2289     struct in_addr alias_addr,
2290     u_char proto)
2291 {
2292 	struct alias_link *lnk;
2293 
2294 	LIBALIAS_LOCK(la);
2295 	lnk = AddLink(la, src_addr, dst_addr, alias_addr,
2296 	    NO_SRC_PORT, NO_DEST_PORT, 0,
2297 	    proto);
2298 
2299 	if (lnk != NULL) {
2300 		lnk->flags |= LINK_PERMANENT;
2301 	}
2302 #ifdef LIBALIAS_DEBUG
2303 	else {
2304 		fprintf(stderr, "PacketAliasRedirectProto(): "
2305 		    "call to AddLink() failed\n");
2306 	}
2307 #endif
2308 
2309 	LIBALIAS_UNLOCK(la);
2310 	return (lnk);
2311 }
2312 
2313 /* Static address translation */
2314 struct alias_link *
2315 LibAliasRedirectAddr(struct libalias *la, struct in_addr src_addr,
2316     struct in_addr alias_addr)
2317 {
2318 	struct alias_link *lnk;
2319 
2320 	LIBALIAS_LOCK(la);
2321 	lnk = AddLink(la, src_addr, la->nullAddress, alias_addr,
2322 	    0, 0, 0,
2323 	    LINK_ADDR);
2324 
2325 	if (lnk != NULL) {
2326 		lnk->flags |= LINK_PERMANENT;
2327 	}
2328 #ifdef LIBALIAS_DEBUG
2329 	else {
2330 		fprintf(stderr, "PacketAliasRedirectAddr(): "
2331 		    "call to AddLink() failed\n");
2332 	}
2333 #endif
2334 
2335 	LIBALIAS_UNLOCK(la);
2336 	return (lnk);
2337 }
2338 
2339 /* Mark the aliasing link dynamic */
2340 int
2341 LibAliasRedirectDynamic(struct libalias *la, struct alias_link *lnk)
2342 {
2343 	int res;
2344 
2345 	LIBALIAS_LOCK(la);
2346 	(void)la;
2347 
2348 	if (lnk->flags & LINK_PARTIALLY_SPECIFIED)
2349 		res = -1;
2350 	else {
2351 		lnk->flags &= ~LINK_PERMANENT;
2352 		res = 0;
2353 	}
2354 	LIBALIAS_UNLOCK(la);
2355 	return (res);
2356 }
2357 
2358 void
2359 LibAliasRedirectDelete(struct libalias *la, struct alias_link *lnk)
2360 {
2361 /* This is a dangerous function to put in the API,
2362    because an invalid pointer can crash the program. */
2363 
2364 	LIBALIAS_LOCK(la);
2365 	la->deleteAllLinks = 1;
2366 	DeleteLink(lnk);
2367 	la->deleteAllLinks = 0;
2368 	LIBALIAS_UNLOCK(la);
2369 }
2370 
2371 void
2372 LibAliasSetAddress(struct libalias *la, struct in_addr addr)
2373 {
2374 
2375 	LIBALIAS_LOCK(la);
2376 	if (la->packetAliasMode & PKT_ALIAS_RESET_ON_ADDR_CHANGE
2377 	    && la->aliasAddress.s_addr != addr.s_addr)
2378 		CleanupAliasData(la);
2379 
2380 	la->aliasAddress = addr;
2381 	LIBALIAS_UNLOCK(la);
2382 }
2383 
2384 void
2385 LibAliasSetTarget(struct libalias *la, struct in_addr target_addr)
2386 {
2387 
2388 	LIBALIAS_LOCK(la);
2389 	la->targetAddress = target_addr;
2390 	LIBALIAS_UNLOCK(la);
2391 }
2392 
2393 static void
2394 finishoff(void)
2395 {
2396 
2397 	while (!LIST_EMPTY(&instancehead))
2398 		LibAliasUninit(LIST_FIRST(&instancehead));
2399 }
2400 
2401 struct libalias *
2402 LibAliasInit(struct libalias *la)
2403 {
2404 	int i;
2405 #ifndef	_KERNEL
2406 	struct timeval tv;
2407 #endif
2408 
2409 	if (la == NULL) {
2410 #ifdef _KERNEL
2411 #undef malloc	/* XXX: ugly */
2412 		la = malloc(sizeof *la, M_ALIAS, M_WAITOK | M_ZERO);
2413 #else
2414 		la = calloc(sizeof *la, 1);
2415 		if (la == NULL)
2416 			return (la);
2417 #endif
2418 
2419 #ifndef	_KERNEL		/* kernel cleans up on module unload */
2420 		if (LIST_EMPTY(&instancehead))
2421 			atexit(finishoff);
2422 #endif
2423 		LIST_INSERT_HEAD(&instancehead, la, instancelist);
2424 
2425 #ifdef	_KERNEL
2426 		la->timeStamp = time_uptime;
2427 		la->lastCleanupTime = time_uptime;
2428 #else
2429 		gettimeofday(&tv, NULL);
2430 		la->timeStamp = tv.tv_sec;
2431 		la->lastCleanupTime = tv.tv_sec;
2432 #endif
2433 
2434 		for (i = 0; i < LINK_TABLE_OUT_SIZE; i++)
2435 			LIST_INIT(&la->linkTableOut[i]);
2436 		for (i = 0; i < LINK_TABLE_IN_SIZE; i++)
2437 			LIST_INIT(&la->linkTableIn[i]);
2438 #ifdef _KERNEL
2439 		AliasSctpInit(la);
2440 #endif
2441 		LIBALIAS_LOCK_INIT(la);
2442 		LIBALIAS_LOCK(la);
2443 	} else {
2444 		LIBALIAS_LOCK(la);
2445 		la->deleteAllLinks = 1;
2446 		CleanupAliasData(la);
2447 		la->deleteAllLinks = 0;
2448 #ifdef _KERNEL
2449 		AliasSctpTerm(la);
2450 		AliasSctpInit(la);
2451 #endif
2452 	}
2453 
2454 	la->aliasAddress.s_addr = INADDR_ANY;
2455 	la->targetAddress.s_addr = INADDR_ANY;
2456 
2457 	la->icmpLinkCount = 0;
2458 	la->udpLinkCount = 0;
2459 	la->tcpLinkCount = 0;
2460 	la->sctpLinkCount = 0;
2461 	la->pptpLinkCount = 0;
2462 	la->protoLinkCount = 0;
2463 	la->fragmentIdLinkCount = 0;
2464 	la->fragmentPtrLinkCount = 0;
2465 	la->sockCount = 0;
2466 
2467 	la->cleanupIndex = 0;
2468 
2469 	la->packetAliasMode = PKT_ALIAS_SAME_PORTS
2470 #ifndef	NO_USE_SOCKETS
2471 	    | PKT_ALIAS_USE_SOCKETS
2472 #endif
2473 	    | PKT_ALIAS_RESET_ON_ADDR_CHANGE;
2474 #ifndef NO_FW_PUNCH
2475 	la->fireWallFD = -1;
2476 #endif
2477 #ifndef _KERNEL
2478 	LibAliasRefreshModules();
2479 #endif
2480 	LIBALIAS_UNLOCK(la);
2481 	return (la);
2482 }
2483 
2484 void
2485 LibAliasUninit(struct libalias *la)
2486 {
2487 
2488 	LIBALIAS_LOCK(la);
2489 #ifdef _KERNEL
2490 	AliasSctpTerm(la);
2491 #endif
2492 	la->deleteAllLinks = 1;
2493 	CleanupAliasData(la);
2494 	la->deleteAllLinks = 0;
2495 	UninitPacketAliasLog(la);
2496 #ifndef NO_FW_PUNCH
2497 	UninitPunchFW(la);
2498 #endif
2499 	LIST_REMOVE(la, instancelist);
2500 	LIBALIAS_UNLOCK(la);
2501 	LIBALIAS_LOCK_DESTROY(la);
2502 	free(la);
2503 }
2504 
2505 /* Change mode for some operations */
2506 unsigned int
2507 LibAliasSetMode(
2508     struct libalias *la,
2509     unsigned int flags,		/* Which state to bring flags to */
2510     unsigned int mask		/* Mask of which flags to affect (use 0 to
2511 				 * do a probe for flag values) */
2512 )
2513 {
2514 	int res = -1;
2515 
2516 	LIBALIAS_LOCK(la);
2517 /* Enable logging? */
2518 	if (flags & mask & PKT_ALIAS_LOG) {
2519 		/* Do the enable */
2520 		if (InitPacketAliasLog(la) == ENOMEM)
2521 			goto getout;
2522 	} else
2523 /* _Disable_ logging? */
2524 	if (~flags & mask & PKT_ALIAS_LOG) {
2525 		UninitPacketAliasLog(la);
2526 	}
2527 #ifndef NO_FW_PUNCH
2528 /* Start punching holes in the firewall? */
2529 	if (flags & mask & PKT_ALIAS_PUNCH_FW) {
2530 		InitPunchFW(la);
2531 	} else
2532 /* Stop punching holes in the firewall? */
2533 	if (~flags & mask & PKT_ALIAS_PUNCH_FW) {
2534 		UninitPunchFW(la);
2535 	}
2536 #endif
2537 
2538 /* Other flags can be set/cleared without special action */
2539 	la->packetAliasMode = (flags & mask) | (la->packetAliasMode & ~mask);
2540 	res = la->packetAliasMode;
2541 getout:
2542 	LIBALIAS_UNLOCK(la);
2543 	return (res);
2544 }
2545 
2546 int
2547 LibAliasCheckNewLink(struct libalias *la)
2548 {
2549 	int res;
2550 
2551 	LIBALIAS_LOCK(la);
2552 	res = la->newDefaultLink;
2553 	LIBALIAS_UNLOCK(la);
2554 	return (res);
2555 }
2556 
2557 #ifndef NO_FW_PUNCH
2558 
2559 /*****************
2560   Code to support firewall punching.  This shouldn't really be in this
2561   file, but making variables global is evil too.
2562   ****************/
2563 
2564 /* Firewall include files */
2565 #include <net/if.h>
2566 #include <netinet/ip_fw.h>
2567 #include <string.h>
2568 #include <err.h>
2569 
2570 /*
2571  * helper function, updates the pointer to cmd with the length
2572  * of the current command, and also cleans up the first word of
2573  * the new command in case it has been clobbered before.
2574  */
2575 static ipfw_insn *
2576 next_cmd(ipfw_insn * cmd)
2577 {
2578 	cmd += F_LEN(cmd);
2579 	bzero(cmd, sizeof(*cmd));
2580 	return (cmd);
2581 }
2582 
2583 /*
2584  * A function to fill simple commands of size 1.
2585  * Existing flags are preserved.
2586  */
2587 static ipfw_insn *
2588 fill_cmd(ipfw_insn * cmd, enum ipfw_opcodes opcode, int size,
2589     int flags, u_int16_t arg)
2590 {
2591 	cmd->opcode = opcode;
2592 	cmd->len = ((cmd->len | flags) & (F_NOT | F_OR)) | (size & F_LEN_MASK);
2593 	cmd->arg1 = arg;
2594 	return next_cmd(cmd);
2595 }
2596 
2597 static ipfw_insn *
2598 fill_ip(ipfw_insn * cmd1, enum ipfw_opcodes opcode, u_int32_t addr)
2599 {
2600 	ipfw_insn_ip *cmd = (ipfw_insn_ip *) cmd1;
2601 
2602 	cmd->addr.s_addr = addr;
2603 	return fill_cmd(cmd1, opcode, F_INSN_SIZE(ipfw_insn_u32), 0, 0);
2604 }
2605 
2606 static ipfw_insn *
2607 fill_one_port(ipfw_insn * cmd1, enum ipfw_opcodes opcode, u_int16_t port)
2608 {
2609 	ipfw_insn_u16 *cmd = (ipfw_insn_u16 *) cmd1;
2610 
2611 	cmd->ports[0] = cmd->ports[1] = port;
2612 	return fill_cmd(cmd1, opcode, F_INSN_SIZE(ipfw_insn_u16), 0, 0);
2613 }
2614 
2615 static int
2616 fill_rule(void *buf, int bufsize, int rulenum,
2617     enum ipfw_opcodes action, int proto,
2618     struct in_addr sa, u_int16_t sp, struct in_addr da, u_int16_t dp)
2619 {
2620 	struct ip_fw *rule = (struct ip_fw *)buf;
2621 	ipfw_insn *cmd = (ipfw_insn *) rule->cmd;
2622 
2623 	bzero(buf, bufsize);
2624 	rule->rulenum = rulenum;
2625 
2626 	cmd = fill_cmd(cmd, O_PROTO, F_INSN_SIZE(ipfw_insn), 0, proto);
2627 	cmd = fill_ip(cmd, O_IP_SRC, sa.s_addr);
2628 	cmd = fill_one_port(cmd, O_IP_SRCPORT, sp);
2629 	cmd = fill_ip(cmd, O_IP_DST, da.s_addr);
2630 	cmd = fill_one_port(cmd, O_IP_DSTPORT, dp);
2631 
2632 	rule->act_ofs = (u_int32_t *) cmd - (u_int32_t *) rule->cmd;
2633 	cmd = fill_cmd(cmd, action, F_INSN_SIZE(ipfw_insn), 0, 0);
2634 
2635 	rule->cmd_len = (u_int32_t *) cmd - (u_int32_t *) rule->cmd;
2636 
2637 	return ((char *)cmd - (char *)buf);
2638 }
2639 
2640 static void	ClearAllFWHoles(struct libalias *la);
2641 
2642 #define fw_setfield(la, field, num)                         \
2643 do {                                                    \
2644     (field)[(num) - la->fireWallBaseNum] = 1;               \
2645 } /*lint -save -e717 */ while(0)/* lint -restore */
2646 
2647 #define fw_clrfield(la, field, num)                         \
2648 do {                                                    \
2649     (field)[(num) - la->fireWallBaseNum] = 0;               \
2650 } /*lint -save -e717 */ while(0)/* lint -restore */
2651 
2652 #define fw_tstfield(la, field, num) ((field)[(num) - la->fireWallBaseNum])
2653 
2654 static void
2655 InitPunchFW(struct libalias *la)
2656 {
2657 
2658 	la->fireWallField = malloc(la->fireWallNumNums);
2659 	if (la->fireWallField) {
2660 		memset(la->fireWallField, 0, la->fireWallNumNums);
2661 		if (la->fireWallFD < 0) {
2662 			la->fireWallFD = socket(AF_INET, SOCK_RAW, IPPROTO_RAW);
2663 		}
2664 		ClearAllFWHoles(la);
2665 		la->fireWallActiveNum = la->fireWallBaseNum;
2666 	}
2667 }
2668 
2669 static void
2670 UninitPunchFW(struct libalias *la)
2671 {
2672 
2673 	ClearAllFWHoles(la);
2674 	if (la->fireWallFD >= 0)
2675 		close(la->fireWallFD);
2676 	la->fireWallFD = -1;
2677 	if (la->fireWallField)
2678 		free(la->fireWallField);
2679 	la->fireWallField = NULL;
2680 	la->packetAliasMode &= ~PKT_ALIAS_PUNCH_FW;
2681 }
2682 
2683 /* Make a certain link go through the firewall */
2684 void
2685 PunchFWHole(struct alias_link *lnk)
2686 {
2687 	struct libalias *la;
2688 	int r;			/* Result code */
2689 	struct ip_fw rule;	/* On-the-fly built rule */
2690 	int fwhole;		/* Where to punch hole */
2691 
2692 	la = lnk->la;
2693 
2694 /* Don't do anything unless we are asked to */
2695 	if (!(la->packetAliasMode & PKT_ALIAS_PUNCH_FW) ||
2696 	    la->fireWallFD < 0 ||
2697 	    lnk->link_type != LINK_TCP)
2698 		return;
2699 
2700 	memset(&rule, 0, sizeof rule);
2701 
2702 /** Build rule **/
2703 
2704 	/* Find empty slot */
2705 	for (fwhole = la->fireWallActiveNum;
2706 	    fwhole < la->fireWallBaseNum + la->fireWallNumNums &&
2707 	    fw_tstfield(la, la->fireWallField, fwhole);
2708 	    fwhole++);
2709 	if (fwhole == la->fireWallBaseNum + la->fireWallNumNums) {
2710 		for (fwhole = la->fireWallBaseNum;
2711 		    fwhole < la->fireWallActiveNum &&
2712 		    fw_tstfield(la, la->fireWallField, fwhole);
2713 		    fwhole++);
2714 		if (fwhole == la->fireWallActiveNum) {
2715 			/* No rule point empty - we can't punch more holes. */
2716 			la->fireWallActiveNum = la->fireWallBaseNum;
2717 #ifdef LIBALIAS_DEBUG
2718 			fprintf(stderr, "libalias: Unable to create firewall hole!\n");
2719 #endif
2720 			return;
2721 		}
2722 	}
2723 	/* Start next search at next position */
2724 	la->fireWallActiveNum = fwhole + 1;
2725 
2726 	/*
2727 	 * generate two rules of the form
2728 	 *
2729 	 * add fwhole accept tcp from OAddr OPort to DAddr DPort add fwhole
2730 	 * accept tcp from DAddr DPort to OAddr OPort
2731 	 */
2732 	if (GetOriginalPort(lnk) != 0 && GetDestPort(lnk) != 0) {
2733 		u_int32_t rulebuf[255];
2734 		int i;
2735 
2736 		i = fill_rule(rulebuf, sizeof(rulebuf), fwhole,
2737 		    O_ACCEPT, IPPROTO_TCP,
2738 		    GetOriginalAddress(lnk), ntohs(GetOriginalPort(lnk)),
2739 		    GetDestAddress(lnk), ntohs(GetDestPort(lnk)));
2740 		r = setsockopt(la->fireWallFD, IPPROTO_IP, IP_FW_ADD, rulebuf, i);
2741 		if (r)
2742 			err(1, "alias punch inbound(1) setsockopt(IP_FW_ADD)");
2743 
2744 		i = fill_rule(rulebuf, sizeof(rulebuf), fwhole,
2745 		    O_ACCEPT, IPPROTO_TCP,
2746 		    GetDestAddress(lnk), ntohs(GetDestPort(lnk)),
2747 		    GetOriginalAddress(lnk), ntohs(GetOriginalPort(lnk)));
2748 		r = setsockopt(la->fireWallFD, IPPROTO_IP, IP_FW_ADD, rulebuf, i);
2749 		if (r)
2750 			err(1, "alias punch inbound(2) setsockopt(IP_FW_ADD)");
2751 	}
2752 
2753 /* Indicate hole applied */
2754 	lnk->data.tcp->fwhole = fwhole;
2755 	fw_setfield(la, la->fireWallField, fwhole);
2756 }
2757 
2758 /* Remove a hole in a firewall associated with a particular alias
2759    lnk.  Calling this too often is harmless. */
2760 static void
2761 ClearFWHole(struct alias_link *lnk)
2762 {
2763 	struct libalias *la;
2764 
2765 	la = lnk->la;
2766 	if (lnk->link_type == LINK_TCP) {
2767 		int fwhole = lnk->data.tcp->fwhole;	/* Where is the firewall
2768 							 * hole? */
2769 		struct ip_fw rule;
2770 
2771 		if (fwhole < 0)
2772 			return;
2773 
2774 		memset(&rule, 0, sizeof rule);	/* useless for ipfw2 */
2775 		while (!setsockopt(la->fireWallFD, IPPROTO_IP, IP_FW_DEL,
2776 		    &fwhole, sizeof fwhole));
2777 		fw_clrfield(la, la->fireWallField, fwhole);
2778 		lnk->data.tcp->fwhole = -1;
2779 	}
2780 }
2781 
2782 /* Clear out the entire range dedicated to firewall holes. */
2783 static void
2784 ClearAllFWHoles(struct libalias *la)
2785 {
2786 	struct ip_fw rule;	/* On-the-fly built rule */
2787 	int i;
2788 
2789 	if (la->fireWallFD < 0)
2790 		return;
2791 
2792 	memset(&rule, 0, sizeof rule);
2793 	for (i = la->fireWallBaseNum; i < la->fireWallBaseNum + la->fireWallNumNums; i++) {
2794 		int r = i;
2795 
2796 		while (!setsockopt(la->fireWallFD, IPPROTO_IP, IP_FW_DEL, &r, sizeof r));
2797 	}
2798 	/* XXX: third arg correct here ? /phk */
2799 	memset(la->fireWallField, 0, la->fireWallNumNums);
2800 }
2801 
2802 #endif /* !NO_FW_PUNCH */
2803 
2804 void
2805 LibAliasSetFWBase(struct libalias *la, unsigned int base, unsigned int num)
2806 {
2807 
2808 	LIBALIAS_LOCK(la);
2809 #ifndef NO_FW_PUNCH
2810 	la->fireWallBaseNum = base;
2811 	la->fireWallNumNums = num;
2812 #endif
2813 	LIBALIAS_UNLOCK(la);
2814 }
2815 
2816 void
2817 LibAliasSetSkinnyPort(struct libalias *la, unsigned int port)
2818 {
2819 
2820 	LIBALIAS_LOCK(la);
2821 	la->skinnyPort = port;
2822 	LIBALIAS_UNLOCK(la);
2823 }
2824 
2825 /*
2826  * Find the address to redirect incoming packets
2827  */
2828 struct in_addr
2829 FindSctpRedirectAddress(struct libalias *la,  struct sctp_nat_msg *sm)
2830 {
2831 	struct alias_link *lnk;
2832 	struct in_addr redir;
2833 
2834 	LIBALIAS_LOCK_ASSERT(la);
2835 	lnk = FindLinkIn(la, sm->ip_hdr->ip_src, sm->ip_hdr->ip_dst,
2836 	    sm->sctp_hdr->dest_port,sm->sctp_hdr->dest_port, LINK_SCTP, 1);
2837 	if (lnk != NULL) {
2838 		return(lnk->src_addr); /* port redirect */
2839 	} else {
2840 		redir = FindOriginalAddress(la,sm->ip_hdr->ip_dst);
2841 		if (redir.s_addr == la->aliasAddress.s_addr ||
2842 		    redir.s_addr == la->targetAddress.s_addr) { /* No address found */
2843 			lnk = FindLinkIn(la, sm->ip_hdr->ip_src, sm->ip_hdr->ip_dst,
2844 			    NO_DEST_PORT, 0, LINK_SCTP, 1);
2845 			if (lnk != NULL)
2846 				return(lnk->src_addr); /* redirect proto */
2847 		}
2848 		return(redir); /* address redirect */
2849 	}
2850 }
2851