1995-11-29 07:40:04 +00:00
|
|
|
/* dhcpd.c
|
|
|
|
|
|
|
|
DHCP Server Daemon. */
|
|
|
|
|
|
|
|
/*
|
1996-02-07 22:43:54 +00:00
|
|
|
* Copyright (c) 1995, 1996 The Internet Software Consortium.
|
|
|
|
* All rights reserved.
|
1995-11-29 07:40:04 +00:00
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
*
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
* 3. Neither the name of The Internet Software Consortium nor the names
|
|
|
|
* of its contributors may be used to endorse or promote products derived
|
|
|
|
* from this software without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE INTERNET SOFTWARE CONSORTIUM AND
|
|
|
|
* CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
|
|
|
|
* INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
|
|
|
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
|
|
* DISCLAIMED. IN NO EVENT SHALL THE INTERNET SOFTWARE CONSORTIUM OR
|
|
|
|
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
|
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
|
|
|
|
* USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
|
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
|
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
|
|
|
|
* OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
|
* SUCH DAMAGE.
|
|
|
|
*
|
|
|
|
* This software has been written for the Internet Software Consortium
|
|
|
|
* by Ted Lemon <mellon@fugue.com> in cooperation with Vixie
|
|
|
|
* Enterprises. To learn more about the Internet Software Consortium,
|
|
|
|
* see ``http://www.vix.com/isc''. To learn more about Vixie
|
|
|
|
* Enterprises, see ``http://www.vix.com''.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef lint
|
|
|
|
static char copyright[] =
|
|
|
|
"@(#) Copyright (c) 1995 The Internet Software Consortium. All rights reserved.\n";
|
|
|
|
#endif /* not lint */
|
|
|
|
|
|
|
|
#include "dhcpd.h"
|
|
|
|
|
|
|
|
static void usage PROTO ((void));
|
|
|
|
|
|
|
|
TIME cur_time;
|
1996-02-21 12:11:09 +00:00
|
|
|
TIME default_lease_time = 43200; /* 12 hours... */
|
|
|
|
TIME max_lease_time = 86400; /* 24 hours... */
|
1995-11-29 07:40:04 +00:00
|
|
|
|
1996-02-21 12:11:09 +00:00
|
|
|
struct subnet *local_subnet;
|
1995-11-29 07:40:04 +00:00
|
|
|
u_int32_t *server_addrlist;
|
|
|
|
int server_addrcount;
|
|
|
|
u_int16_t server_port;
|
1996-02-21 15:16:18 +00:00
|
|
|
struct iaddr siaddr;
|
1995-11-29 07:40:04 +00:00
|
|
|
|
|
|
|
int main (argc, argv, envp)
|
|
|
|
int argc;
|
|
|
|
char **argv, **envp;
|
|
|
|
{
|
|
|
|
struct in_addr addr;
|
|
|
|
int port = 0;
|
|
|
|
int i;
|
|
|
|
struct sockaddr_in name;
|
|
|
|
u_int32_t *addrlist = (u_int32_t *)0;
|
|
|
|
int addrcount = 0;
|
|
|
|
struct tree *addrtree = (struct tree *)0;
|
|
|
|
struct servent *ent;
|
|
|
|
int sock;
|
|
|
|
int pid;
|
|
|
|
int result;
|
|
|
|
int flag;
|
|
|
|
|
|
|
|
openlog ("dhcpd", LOG_NDELAY, LOG_DAEMON);
|
|
|
|
#ifndef DEBUG
|
|
|
|
setlogmask (LOG_UPTO (LOG_INFO));
|
|
|
|
|
|
|
|
/* Become a daemon... */
|
|
|
|
if ((pid = fork ()) < 0)
|
|
|
|
error ("Can't fork daemon: %m");
|
|
|
|
else if (pid)
|
|
|
|
exit (0);
|
|
|
|
/* Become session leader and get pid... */
|
|
|
|
pid = setsid ();
|
|
|
|
#endif
|
|
|
|
for (i = 1; i < argc; i++) {
|
|
|
|
if (!strcmp (argv [i], "-p")) {
|
|
|
|
if (++i == argc)
|
|
|
|
usage ();
|
|
|
|
server_port = htons (atoi (argv [i]));
|
1996-02-29 18:18:40 +00:00
|
|
|
debug ("binding to user-specified port %d",
|
1995-11-29 07:40:04 +00:00
|
|
|
ntohs (server_port));
|
|
|
|
#if 0
|
|
|
|
} else if (!strcmp (argv [i], "-a")) {
|
|
|
|
if (++i == argc)
|
|
|
|
usage ();
|
|
|
|
if (inet_aton (argv [i], &addr)) {
|
|
|
|
addrtree =
|
|
|
|
tree_concat (addrtree,
|
|
|
|
tree_const
|
|
|
|
((unsigned char *)&addr,
|
|
|
|
sizeof addr));
|
|
|
|
} else {
|
|
|
|
addrtree = tree_concat (addrtree,
|
|
|
|
tree_host_lookup
|
|
|
|
(argv [i]));
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
} else
|
|
|
|
usage ();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Default to the DHCP/BOOTP port. */
|
|
|
|
if (!server_port)
|
|
|
|
{
|
|
|
|
ent = getservbyname ("dhcp", "udp");
|
|
|
|
if (!ent)
|
|
|
|
server_port = htons (67);
|
|
|
|
else
|
|
|
|
server_port = ent -> s_port;
|
|
|
|
endservent ();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get the current time... */
|
|
|
|
GET_TIME (&cur_time);
|
|
|
|
|
|
|
|
/* Read the dhcpd.conf file... */
|
|
|
|
readconf ();
|
|
|
|
|
1996-03-02 05:13:36 +00:00
|
|
|
/* Start up the database... */
|
|
|
|
db_startup ();
|
|
|
|
|
1995-11-29 07:40:04 +00:00
|
|
|
#if 0
|
|
|
|
/* If addresses were specified on the command line, resolve them;
|
|
|
|
otherwise, just get a list of the addresses that are configured
|
|
|
|
on this host and listen on all of them. */
|
|
|
|
if (addrtree) {
|
|
|
|
tree_evaluate ((unsigned char **)&addrlist,
|
|
|
|
&addrcount, addrtree);
|
|
|
|
addrcount /= 4;
|
|
|
|
if (!addrcount)
|
|
|
|
error ("Server addresses resolve to nothing.");
|
|
|
|
} else {
|
|
|
|
/* addrlist = get_interface_list (&addrcount); */
|
|
|
|
#endif
|
|
|
|
addr.s_addr = 0;
|
|
|
|
addrlist = (u_int32_t *)&(addr.s_addr);
|
|
|
|
addrcount = 1;
|
|
|
|
#if 0
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
1996-02-21 15:16:18 +00:00
|
|
|
siaddr.len = 0;
|
1995-11-29 07:40:04 +00:00
|
|
|
server_addrlist = get_interface_list (&server_addrcount);
|
1996-02-21 12:11:09 +00:00
|
|
|
for (i = 0; i < server_addrcount; i++) {
|
|
|
|
struct sockaddr_in foo;
|
|
|
|
foo.sin_addr.s_addr = server_addrlist [i];
|
|
|
|
|
|
|
|
if (server_addrlist [i] != htonl (INADDR_LOOPBACK)) {
|
1996-02-21 15:16:18 +00:00
|
|
|
if (siaddr.len) {
|
1996-02-21 12:11:09 +00:00
|
|
|
error ("dhcpd currently does not support "
|
|
|
|
"multiple interfaces");
|
|
|
|
}
|
1996-02-21 15:16:18 +00:00
|
|
|
siaddr.len = 4;
|
|
|
|
memcpy (siaddr.iabuf, &server_addrlist [i], 4);
|
|
|
|
local_subnet = find_subnet (siaddr);
|
1996-02-21 12:11:09 +00:00
|
|
|
}
|
|
|
|
}
|
1995-11-29 07:40:04 +00:00
|
|
|
|
|
|
|
/* Listen on the specified (or default) port on each specified
|
|
|
|
(or default) IP address. */
|
|
|
|
for (i = 0; i < addrcount; i++) {
|
|
|
|
listen_on (server_port, addrlist [i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Write a pid file. */
|
|
|
|
if ((i = open (_PATH_DHCPD_PID, O_WRONLY | O_CREAT)) >= 0) {
|
|
|
|
char obuf [20];
|
|
|
|
sprintf (obuf, "%d\n", getpid ());
|
|
|
|
write (i, obuf, strlen (obuf));
|
|
|
|
close (i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Receive packets and dispatch them... */
|
|
|
|
dispatch ();
|
1996-02-06 20:25:56 +00:00
|
|
|
|
|
|
|
/* Not reached */
|
|
|
|
return 0;
|
1995-11-29 07:40:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Print usage message. */
|
|
|
|
|
|
|
|
static void usage ()
|
|
|
|
{
|
|
|
|
error ("Usage: dhcpd [-p <port>] [-a <ip-addr>]");
|
|
|
|
}
|
|
|
|
|
|
|
|
void cleanup ()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
1996-02-06 20:25:56 +00:00
|
|
|
void do_packet (packbuf, len, from_port, from, sock)
|
1995-11-29 07:40:04 +00:00
|
|
|
unsigned char *packbuf;
|
|
|
|
int len;
|
1996-02-06 20:25:56 +00:00
|
|
|
unsigned long from_port;
|
|
|
|
struct iaddr from;
|
1995-11-29 07:40:04 +00:00
|
|
|
int sock;
|
|
|
|
{
|
|
|
|
struct packet *tp;
|
|
|
|
struct dhcp_packet *tdp;
|
1996-02-21 12:11:09 +00:00
|
|
|
struct iaddr ia;
|
1995-11-29 07:40:04 +00:00
|
|
|
|
|
|
|
if (!(tp = new_packet ("do_packet")))
|
|
|
|
return;
|
|
|
|
if (!(tdp = new_dhcp_packet ("do_packet"))) {
|
|
|
|
free_packet (tp, "do_packet");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
memcpy (tdp, packbuf, len);
|
|
|
|
memset (tp, 0, sizeof *tp);
|
|
|
|
tp -> raw = tdp;
|
|
|
|
tp -> packet_length = len;
|
1996-02-06 20:25:56 +00:00
|
|
|
tp -> client_port = from_port;
|
|
|
|
tp -> client_addr = from;
|
1995-11-29 07:40:04 +00:00
|
|
|
tp -> client_sock = sock;
|
1996-02-21 12:11:09 +00:00
|
|
|
|
|
|
|
/* If this came through a gateway, find the corresponding subnet... */
|
|
|
|
if (tp -> raw -> giaddr.s_addr) {
|
|
|
|
ia.len = 4;
|
|
|
|
memcpy (ia.iabuf, &tp -> raw -> giaddr, 4);
|
|
|
|
tp -> subnet = find_subnet (ia);
|
|
|
|
} else {
|
|
|
|
tp -> subnet = local_subnet;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If the subnet from whence this packet came is unknown to us,
|
|
|
|
drop it on the floor... */
|
|
|
|
if (!tp -> subnet)
|
|
|
|
note ("Packet from unknown subnet: %s",
|
|
|
|
inet_ntoa (tp -> raw -> giaddr));
|
|
|
|
else {
|
|
|
|
parse_options (tp);
|
|
|
|
if (tp -> options_valid &&
|
|
|
|
tp -> options [DHO_DHCP_MESSAGE_TYPE].data)
|
1996-02-26 01:56:39 +00:00
|
|
|
tp -> packet_type =
|
|
|
|
tp -> options [DHO_DHCP_MESSAGE_TYPE].data [0];
|
|
|
|
if (tp -> packet_type)
|
1996-02-21 12:11:09 +00:00
|
|
|
dhcp (tp);
|
|
|
|
else
|
|
|
|
bootp (tp);
|
|
|
|
}
|
1995-11-29 07:40:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Based on the contents of packet, figure out which interface address
|
|
|
|
to use from server_addrlist. Currently just picks the first
|
|
|
|
interface. */
|
|
|
|
|
|
|
|
u_int32_t pick_interface (packet)
|
|
|
|
struct packet *packet;
|
|
|
|
{
|
|
|
|
if (server_addrlist)
|
|
|
|
return server_addrlist [0];
|
|
|
|
return 0;
|
|
|
|
}
|