⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 dhcrelay.c

📁 DHCP服务器源码
💻 C
📖 第 1 页 / 共 2 页
字号:
/* dhcrelay.c   DHCP/BOOTP Relay Agent. *//* * Copyright (c) 2004 by Internet Systems Consortium, Inc. ("ISC") * Copyright (c) 1997-2003 by Internet Software Consortium * * Permission to use, copy, modify, and distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS.  IN NO EVENT SHALL ISC BE LIABLE FOR * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT * OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. * *   Internet Systems Consortium, Inc. *   950 Charter Street *   Redwood City, CA 94063 *   <info@isc.org> *   http://www.isc.org/ * * This software has been written for Internet Systems Consortium * by Ted Lemon in cooperation with Vixie Enterprises and Nominum, Inc. * To learn more about Internet Systems Consortium, see * ``http://www.isc.org/''.  To learn more about Vixie Enterprises, * see ``http://www.vix.com''.   To learn more about Nominum, Inc., see * ``http://www.nominum.com''. */#ifndef lintstatic char ocopyright[] ="$Id: dhcrelay.c,v 1.52.2.7 2004/07/10 00:11:17 dhankins Exp $ Copyright (c) 2004 Internet Systems Consortium.  All rights reserved.\n";#endif /* not lint */#include "dhcpd.h"#include "version.h"static void usage PROTO ((void));TIME default_lease_time = 43200; /* 12 hours... */TIME max_lease_time = 86400; /* 24 hours... */struct tree_cache *global_options [256];/* Needed to prevent linking against conflex.c. */int lexline;int lexchar;char *token_line;char *tlname;const char *path_dhcrelay_pid = _PATH_DHCRELAY_PID;int bogus_agent_drops = 0;	/* Packets dropped because agent option				   field was specified and we're not relaying				   packets that already have an agent option				   specified. */int bogus_giaddr_drops = 0;	/* Packets sent to us to relay back to a				   client, but with a bogus giaddr. */int client_packets_relayed = 0;	/* Packets relayed from client to server. */int server_packet_errors = 0;	/* Errors sending packets to servers. */int server_packets_relayed = 0;	/* Packets relayed from server to client. */int client_packet_errors = 0;	/* Errors sending packets to clients. */int add_agent_options = 0;	/* If nonzero, add relay agent options. */int drop_agent_mismatches = 0;	/* If nonzero, drop server replies that				   don't contain a Relay Agent Information				   option whose Agent ID suboption matches				   our giaddr. */int corrupt_agent_options = 0;	/* Number of packets dropped because				   relay agent information option was bad. */int missing_agent_option = 0;	/* Number of packets dropped because no				   RAI option matching our ID was found. */int bad_circuit_id = 0;		/* Circuit ID option in matching RAI option				   did not match any known circuit ID. */int missing_circuit_id = 0;	/* Circuit ID option in matching RAI option				   was missing. */int max_hop_count = 10;		/* Maximum hop count */	/* Maximum size of a packet with agent options added. */int dhcp_max_agent_option_packet_length = 576;	/* What to do about packets we're asked to relay that	   already have a relay option: */enum { forward_and_append,	/* Forward and append our own relay option. */       forward_and_replace,	/* Forward, but replace theirs with ours. */       forward_untouched,	/* Forward without changes. */       discard } agent_relay_mode = forward_and_replace;u_int16_t local_port;u_int16_t remote_port;struct server_list {	struct server_list *next;	struct sockaddr_in to;} *servers;static char copyright [] = "Copyright 2004 Internet Systems Consortium.";static char arr [] = "All rights reserved.";static char message [] = "Internet Systems Consortium DHCP Relay Agent";static char url [] = "For info, please visit http://www.isc.org/sw/dhcp/";int main (argc, argv, envp)	int argc;	char **argv, **envp;{	int i;	struct servent *ent;	struct server_list *sp = (struct server_list *)0;	int no_daemon = 0;	int quiet = 0;	isc_result_t status;	char *s;	/* Make sure we have stdin, stdout and stderr. */	i = open ("/dev/null", O_RDWR);	if (i == 0)		i = open ("/dev/null", O_RDWR);	if (i == 1) {		i = open ("/dev/null", O_RDWR);		log_perror = 0; /* No sense logging to /dev/null. */	} else if (i != -1)		close (i);#ifdef SYSLOG_4_2	openlog ("dhcrelay", LOG_NDELAY);	log_priority = LOG_DAEMON;#else	openlog ("dhcrelay", LOG_NDELAY, LOG_DAEMON);#endif#if !(defined (DEBUG) || defined (SYSLOG_4_2))	setlogmask (LOG_UPTO (LOG_INFO));#endif		/* Set up the OMAPI. */	status = omapi_init ();	if (status != ISC_R_SUCCESS)		log_fatal ("Can't initialize OMAPI: %s",			   isc_result_totext (status));	/* Set up the OMAPI wrappers for the interface object. */	interface_setup ();	for (i = 1; i < argc; i++) {		if (!strcmp (argv [i], "-p")) {			if (++i == argc)				usage ();			local_port = htons (atoi (argv [i]));			log_debug ("binding to user-specified port %d",			       ntohs (local_port));		} else if (!strcmp (argv [i], "-d")) {			no_daemon = 1; 		} else if (!strcmp (argv [i], "-i")) {			struct interface_info *tmp =				(struct interface_info *)0;			status = interface_allocate (&tmp, MDL);			if (status != ISC_R_SUCCESS)				log_fatal ("%s: interface_allocate: %s",					   argv [i],					   isc_result_totext (status));			if (++i == argc) {				usage ();			}			strcpy (tmp -> name, argv [i]);			interface_snorf (tmp, INTERFACE_REQUESTED);			interface_dereference (&tmp, MDL);		} else if (!strcmp (argv [i], "-q")) {			quiet = 1;			quiet_interface_discovery = 1;		} else if (!strcmp (argv [i], "-a")) {			add_agent_options = 1;		} else if (!strcmp (argv [i], "-c")) {			int hcount;			if (++i == argc)				usage ();			hcount = atoi(argv[i]);			if (hcount <= 255)				max_hop_count= hcount;			else				usage ();		} else if (!strcmp (argv [i], "-A")) {			if (++i == argc)				usage ();			dhcp_max_agent_option_packet_length = atoi (argv [i]);		} else if (!strcmp (argv [i], "-m")) {			if (++i == argc)				usage ();			if (!strcasecmp (argv [i], "append")) {				agent_relay_mode = forward_and_append;			} else if (!strcasecmp (argv [i], "replace")) {				agent_relay_mode = forward_and_replace;			} else if (!strcasecmp (argv [i], "forward")) {				agent_relay_mode = forward_untouched;			} else if (!strcasecmp (argv [i], "discard")) {				agent_relay_mode = discard;			} else				usage ();		} else if (!strcmp (argv [i], "-D")) {			drop_agent_mismatches = 1; 		} else if (argv [i][0] == '-') { 		    usage ();		} else if (!strcmp (argv [i], "--version")) {			log_info ("isc-dhcrelay-%s", DHCP_VERSION);			exit (0); 		} else {			struct hostent *he;			struct in_addr ia, *iap = (struct in_addr *)0;			if (inet_aton (argv [i], &ia)) {				iap = &ia;			} else {				he = gethostbyname (argv [i]);				if (!he) {					log_error ("%s: host unknown",						   argv [i]);				} else {					iap = ((struct in_addr *)					       he -> h_addr_list [0]);				}			}			if (iap) {				sp = ((struct server_list *)				      dmalloc (sizeof *sp, MDL));				if (!sp)					log_fatal ("no memory for server.\n");				sp -> next = servers;				servers = sp;				memcpy (&sp -> to.sin_addr,					iap, sizeof *iap);			} 		}	}	if ((s = getenv ("PATH_DHCRELAY_PID"))) {		path_dhcrelay_pid = s;	}	if (!quiet) {		log_info ("%s %s", message, DHCP_VERSION);		log_info (copyright);		log_info (arr);		log_info (url);	} else {		quiet = 0;		log_perror = 0;	}	/* Default to the DHCP/BOOTP port. */	if (!local_port) {		ent = getservbyname ("dhcps", "udp");		if (!ent)			local_port = htons (67);		else			local_port = ent -> s_port;		endservent ();	}	remote_port = htons (ntohs (local_port) + 1);  	/* We need at least one server. */	if (!sp) {		usage ();	}	/* Set up the server sockaddrs. */	for (sp = servers; sp; sp = sp -> next) {		sp -> to.sin_port = local_port;		sp -> to.sin_family = AF_INET;#ifdef HAVE_SA_LEN		sp -> to.sin_len = sizeof sp -> to;#endif	}	/* Get the current time... */	GET_TIME (&cur_time);	/* Discover all the network interfaces. */	discover_interfaces (DISCOVER_RELAY);	/* Set up the bootp packet handler... */	bootp_packet_handler = relay;	/* Become a daemon... */	if (!no_daemon) {		int pid;		FILE *pf;		int pfdesc;		log_perror = 0;		if ((pid = fork()) < 0)			log_fatal ("can't fork daemon: %m");		else if (pid)			exit (0);		pfdesc = open (path_dhcrelay_pid,			       O_CREAT | O_TRUNC | O_WRONLY, 0644);		if (pfdesc < 0) {			log_error ("Can't create %s: %m", path_dhcrelay_pid);		} else {			pf = fdopen (pfdesc, "w");			if (!pf)				log_error ("Can't fdopen %s: %m",				      path_dhcrelay_pid);			else {				fprintf (pf, "%ld\n", (long)getpid ());				fclose (pf);			}			}		close (0);		close (1);		close (2);		pid = setsid ();	}	/* Start dispatching packets and timeouts... */	dispatch ();	/*NOTREACHED*/	return 0;}void relay (ip, packet, length, from_port, from, hfrom)	struct interface_info *ip;	struct dhcp_packet *packet;	unsigned length;	unsigned int from_port;	struct iaddr from;	struct hardware *hfrom;{	struct server_list *sp;	struct sockaddr_in to;	struct interface_info *out;	struct hardware hto, *htop;	if (packet -> hlen > sizeof packet -> chaddr) {		log_info ("Discarding packet with invalid hlen.");		return;	}	/* Find the interface that corresponds to the giaddr	   in the packet. */	if (packet -> giaddr.s_addr) {		for (out = interfaces; out; out = out -> next) {			if (!memcmp (&out -> primary_address,				     &packet -> giaddr,				     sizeof packet -> giaddr))				break;		}	} else {		out = (struct interface_info *)0;	}	/* If it's a bootreply, forward it to the client. */	if (packet -> op == BOOTREPLY) {		if (!(packet -> flags & htons (BOOTP_BROADCAST)) &&			can_unicast_without_arp (out)) {			to.sin_addr = packet -> yiaddr;			to.sin_port = remote_port;			/* and hardware address is not broadcast */			htop = &hto;		} else {			to.sin_addr.s_addr = htonl (INADDR_BROADCAST);			to.sin_port = remote_port;			/* hardware address is broadcast */			htop = NULL;		}		to.sin_family = AF_INET;#ifdef HAVE_SA_LEN		to.sin_len = sizeof to;#endif		memcpy (&hto.hbuf [1], packet -> chaddr, packet -> hlen);		hto.hbuf [0] = packet -> htype;		hto.hlen = packet -> hlen + 1;		/* Wipe out the agent relay options and, if possible, figure		   out which interface to use based on the contents of the		   option that we put on the request to which the server is		   replying. */		if (!(length =		      strip_relay_agent_options (ip, &out, packet, length)))			return;		if (!out) {			log_error ("packet to bogus giaddr %s.\n",			      inet_ntoa (packet -> giaddr));			++bogus_giaddr_drops;			return;		}		if (send_packet (out,				 (struct packet *)0,				 packet, length, out -> primary_address,				 &to, htop) < 0) {			++server_packet_errors;		} else {			log_debug ("forwarded BOOTREPLY for %s to %s",			       print_hw_addr (packet -> htype, packet -> hlen,					      packet -> chaddr),			       inet_ntoa (to.sin_addr));			++server_packets_relayed;		}		return;	}	/* If giaddr matches one of our addresses, ignore the packet -	   we just sent it. */	if (out)		return;	/* Add relay agent options if indicated.   If something goes wrong,	   drop the packet. */	if (!(length = add_relay_agent_options (ip, packet, length,						ip -> primary_address)))		return;	/* If giaddr is not already set, Set it so the server can	   figure out what net it's from and so that we can later	   forward the response to the correct net.    If it's already	   set, the response will be sent directly to the relay agent	   that set giaddr, so we won't see it. */	if (!packet -> giaddr.s_addr)		packet -> giaddr = ip -> primary_address;	if (packet -> hops < max_hop_count)

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -