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