2024-08-21 03:35:12 -07:00
|
|
|
/* $OpenBSD: dhcrelay.c,v 1.67 2024/08/21 10:35:12 florian Exp $ */
|
2004-04-13 16:16:17 -07:00
|
|
|
|
2004-04-12 14:10:28 -07:00
|
|
|
/*
|
2004-04-12 15:25:38 -07:00
|
|
|
* Copyright (c) 2004 Henning Brauer <henning@cvs.openbsd.org>
|
2004-04-12 14:10:28 -07:00
|
|
|
* Copyright (c) 1997, 1998, 1999 The Internet Software Consortium.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* 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''.
|
|
|
|
*/
|
|
|
|
|
2016-02-06 16:49:28 -08:00
|
|
|
#include <sys/types.h>
|
2009-11-03 02:14:09 -08:00
|
|
|
#include <sys/ioctl.h>
|
2016-02-06 16:49:28 -08:00
|
|
|
#include <sys/socket.h>
|
|
|
|
|
|
|
|
#include <arpa/inet.h>
|
|
|
|
|
|
|
|
#include <net/if.h>
|
|
|
|
|
|
|
|
#include <errno.h>
|
2016-09-15 09:16:03 -07:00
|
|
|
#include <fcntl.h>
|
2016-02-06 16:49:28 -08:00
|
|
|
#include <netdb.h>
|
|
|
|
#include <paths.h>
|
|
|
|
#include <pwd.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <syslog.h>
|
2016-08-26 18:26:22 -07:00
|
|
|
#include <time.h>
|
2016-02-06 16:49:28 -08:00
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#include "dhcp.h"
|
|
|
|
#include "dhcpd.h"
|
2017-02-13 11:15:39 -08:00
|
|
|
#include "log.h"
|
2004-04-12 14:10:28 -07:00
|
|
|
|
2004-04-12 14:31:14 -07:00
|
|
|
void usage(void);
|
2016-09-15 09:16:03 -07:00
|
|
|
int rdaemon(int);
|
2004-04-12 14:10:28 -07:00
|
|
|
void relay(struct interface_info *, struct dhcp_packet *, int,
|
2016-12-08 11:18:15 -08:00
|
|
|
struct packet_ctx *);
|
2016-12-12 07:41:05 -08:00
|
|
|
void l2relay(struct interface_info *, struct dhcp_packet *, int,
|
|
|
|
struct packet_ctx *);
|
2004-04-12 14:10:28 -07:00
|
|
|
char *print_hw_addr(int, int, unsigned char *);
|
2004-10-12 09:39:35 -07:00
|
|
|
void got_response(struct protocol *);
|
2009-11-03 02:14:09 -08:00
|
|
|
int get_rdomain(char *);
|
2004-04-12 14:10:28 -07:00
|
|
|
|
2016-12-13 07:28:19 -08:00
|
|
|
void relay_agentinfo(struct packet_ctx *, struct interface_info *, int);
|
2016-12-12 07:41:05 -08:00
|
|
|
|
|
|
|
int relay_agentinfo_cmp(struct packet_ctx *pc, uint8_t *, int);
|
|
|
|
ssize_t relay_agentinfo_append(struct packet_ctx *, struct dhcp_packet *,
|
|
|
|
size_t);
|
|
|
|
ssize_t relay_agentinfo_remove(struct packet_ctx *, struct dhcp_packet *,
|
|
|
|
size_t);
|
2009-09-03 04:56:49 -07:00
|
|
|
|
2004-04-12 14:10:28 -07:00
|
|
|
time_t cur_time;
|
|
|
|
|
|
|
|
struct interface_info *interfaces = NULL;
|
2017-03-16 02:17:20 -07:00
|
|
|
struct server_list *servers;
|
2017-03-15 07:31:49 -07:00
|
|
|
struct iflist intflist;
|
2009-09-03 04:56:49 -07:00
|
|
|
int server_fd;
|
|
|
|
int oflag;
|
2004-04-12 14:10:28 -07:00
|
|
|
|
2016-12-12 07:41:05 -08:00
|
|
|
enum dhcp_relay_mode drm = DRM_UNKNOWN;
|
|
|
|
const char *rai_circuit = NULL;
|
|
|
|
const char *rai_remote = NULL;
|
2016-12-16 10:38:39 -08:00
|
|
|
int rai_replace = 0;
|
2016-12-12 07:41:05 -08:00
|
|
|
|
2004-04-12 15:25:38 -07:00
|
|
|
int
|
|
|
|
main(int argc, char *argv[])
|
2004-04-12 14:10:28 -07:00
|
|
|
{
|
2016-09-15 09:16:03 -07:00
|
|
|
int ch, devnull = -1, daemonize, opt, rdomain;
|
2004-04-12 15:01:37 -07:00
|
|
|
struct server_list *sp = NULL;
|
2004-04-12 18:22:30 -07:00
|
|
|
struct passwd *pw;
|
2004-10-12 09:39:35 -07:00
|
|
|
struct sockaddr_in laddr;
|
2016-12-12 07:41:05 -08:00
|
|
|
int optslen;
|
2004-04-12 14:10:28 -07:00
|
|
|
|
2016-09-04 03:43:52 -07:00
|
|
|
daemonize = 1;
|
|
|
|
|
2017-02-13 11:15:39 -08:00
|
|
|
log_init(1, LOG_DAEMON); /* log to stderr until daemonized */
|
2004-04-12 14:10:28 -07:00
|
|
|
|
2017-03-15 07:31:49 -07:00
|
|
|
setup_iflist();
|
|
|
|
|
2016-12-16 10:38:39 -08:00
|
|
|
while ((ch = getopt(argc, argv, "aC:di:oR:r")) != -1) {
|
2004-04-12 15:01:37 -07:00
|
|
|
switch (ch) {
|
2016-12-12 07:41:05 -08:00
|
|
|
case 'C':
|
|
|
|
rai_circuit = optarg;
|
|
|
|
break;
|
2004-04-12 15:01:37 -07:00
|
|
|
case 'd':
|
2016-09-04 03:43:52 -07:00
|
|
|
daemonize = 0;
|
2004-04-12 15:01:37 -07:00
|
|
|
break;
|
|
|
|
case 'i':
|
|
|
|
if (interfaces != NULL)
|
2004-04-12 14:31:14 -07:00
|
|
|
usage();
|
2016-12-07 05:19:18 -08:00
|
|
|
|
2017-03-15 07:31:49 -07:00
|
|
|
interfaces = iflist_getbyname(optarg);
|
2016-12-08 01:29:50 -08:00
|
|
|
if (interfaces == NULL)
|
2017-02-13 11:15:39 -08:00
|
|
|
fatalx("interface '%s' not found", optarg);
|
2004-04-12 15:01:37 -07:00
|
|
|
break;
|
2009-09-03 04:56:49 -07:00
|
|
|
case 'o':
|
|
|
|
/* add the relay agent information option */
|
|
|
|
oflag++;
|
|
|
|
break;
|
2016-12-12 07:41:05 -08:00
|
|
|
case 'R':
|
|
|
|
rai_remote = optarg;
|
|
|
|
break;
|
2016-12-16 10:38:39 -08:00
|
|
|
case 'r':
|
|
|
|
rai_replace = 1;
|
|
|
|
break;
|
2016-12-07 05:19:18 -08:00
|
|
|
|
2004-04-12 15:01:37 -07:00
|
|
|
default:
|
|
|
|
usage();
|
|
|
|
/* not reached */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
argc -= optind;
|
|
|
|
argv += optind;
|
|
|
|
|
|
|
|
if (argc < 1)
|
|
|
|
usage();
|
|
|
|
|
2016-12-12 07:41:05 -08:00
|
|
|
if (rai_remote != NULL && rai_circuit == NULL)
|
2017-02-13 11:15:39 -08:00
|
|
|
fatalx("you must specify a circuit-id with a remote-id");
|
2016-12-12 07:41:05 -08:00
|
|
|
|
|
|
|
/* Validate that we have space for all suboptions. */
|
|
|
|
if (rai_circuit != NULL) {
|
|
|
|
optslen = 2 + strlen(rai_circuit);
|
|
|
|
if (rai_remote != NULL)
|
|
|
|
optslen += 2 + strlen(rai_remote);
|
|
|
|
|
|
|
|
if (optslen > DHCP_OPTION_MAXLEN)
|
2017-02-13 11:15:39 -08:00
|
|
|
fatalx("relay agent information is too long");
|
2016-12-12 07:41:05 -08:00
|
|
|
}
|
|
|
|
|
2004-04-12 15:01:37 -07:00
|
|
|
while (argc > 0) {
|
2024-08-21 03:35:12 -07:00
|
|
|
struct addrinfo hints, *res;
|
2004-04-12 15:01:37 -07:00
|
|
|
struct in_addr ia, *iap = NULL;
|
|
|
|
|
2016-12-12 07:41:05 -08:00
|
|
|
if ((sp = calloc(1, sizeof(*sp))) == NULL)
|
2017-02-13 11:15:39 -08:00
|
|
|
fatalx("calloc");
|
2016-12-12 07:41:05 -08:00
|
|
|
|
2024-08-21 03:35:12 -07:00
|
|
|
memset(&hints, 0, sizeof(hints));
|
|
|
|
hints.ai_family = AF_INET;
|
|
|
|
|
2017-03-15 07:31:49 -07:00
|
|
|
if ((sp->intf = register_interface(argv[0], got_one,
|
|
|
|
1)) != NULL) {
|
2016-12-12 07:41:05 -08:00
|
|
|
if (drm == DRM_LAYER3)
|
2017-02-13 11:15:39 -08:00
|
|
|
fatalx("don't mix interfaces with hosts");
|
2016-12-12 07:41:05 -08:00
|
|
|
|
|
|
|
if (sp->intf->hw_address.htype == HTYPE_IPSEC_TUNNEL)
|
2018-03-16 05:31:09 -07:00
|
|
|
fatalx("can't use IPsec with layer 2");
|
2016-12-12 07:41:05 -08:00
|
|
|
|
|
|
|
sp->next = servers;
|
|
|
|
servers = sp;
|
|
|
|
|
|
|
|
drm = DRM_LAYER2;
|
|
|
|
argc--;
|
|
|
|
argv++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2024-08-21 03:35:12 -07:00
|
|
|
if (getaddrinfo(argv[0], NULL, &hints, &res) != 0)
|
|
|
|
log_warnx("%s: host unknown", argv[0]);
|
2004-04-12 15:01:37 -07:00
|
|
|
else {
|
2024-08-21 03:35:12 -07:00
|
|
|
ia = ((struct sockaddr_in *)res->ai_addr)->sin_addr;
|
|
|
|
iap = &ia;
|
|
|
|
freeaddrinfo(res);
|
2004-04-12 15:01:37 -07:00
|
|
|
}
|
2024-08-21 03:35:12 -07:00
|
|
|
|
2004-04-12 15:01:37 -07:00
|
|
|
if (iap) {
|
2016-12-12 07:41:05 -08:00
|
|
|
if (drm == DRM_LAYER2)
|
2017-02-13 11:15:39 -08:00
|
|
|
fatalx("don't mix interfaces with hosts");
|
2016-12-12 07:41:05 -08:00
|
|
|
|
|
|
|
drm = DRM_LAYER3;
|
2004-04-12 15:25:38 -07:00
|
|
|
sp->next = servers;
|
2004-04-12 15:01:37 -07:00
|
|
|
servers = sp;
|
2017-03-16 02:17:20 -07:00
|
|
|
memcpy(&ss2sin(&sp->to)->sin_addr, iap, sizeof(*iap));
|
2016-12-12 07:41:05 -08:00
|
|
|
} else
|
|
|
|
free(sp);
|
|
|
|
|
2004-04-12 15:01:37 -07:00
|
|
|
argc--;
|
|
|
|
argv++;
|
2004-04-12 14:10:28 -07:00
|
|
|
}
|
|
|
|
|
2016-09-15 09:16:03 -07:00
|
|
|
if (daemonize) {
|
For open/openat, if the flags parameter does not contain O_CREAT, the
3rd (variadic) mode_t parameter is irrelevant. Many developers in the past
have passed mode_t (0, 044, 0644, or such), which might lead future people
to copy this broken idiom, and perhaps even believe this parameter has some
meaning or implication or application. Delete them all.
This comes out of a conversation where tb@ noticed that a strange (but
intentional) pledge behaviour is to always knock-out high-bits from
mode_t on a number of system calls as a safety factor, and his bewilderment
that this appeared to be happening against valid modes (at least visually),
but no sorry, they are all irrelevant junk. They could all be 0xdeafbeef.
ok millert
2021-10-24 14:24:15 -07:00
|
|
|
devnull = open(_PATH_DEVNULL, O_RDWR);
|
2016-09-15 09:16:03 -07:00
|
|
|
if (devnull == -1)
|
2017-02-13 14:49:38 -08:00
|
|
|
fatal("open(%s)", _PATH_DEVNULL);
|
2016-09-15 09:16:03 -07:00
|
|
|
}
|
2004-04-12 14:10:28 -07:00
|
|
|
|
2017-03-15 07:31:49 -07:00
|
|
|
if (interfaces == NULL ||
|
|
|
|
register_interface(interfaces->name, got_one, 0) == NULL)
|
2017-02-13 11:15:39 -08:00
|
|
|
fatalx("no interface given");
|
2017-03-15 07:31:49 -07:00
|
|
|
|
2016-12-12 07:41:05 -08:00
|
|
|
/* We need an address for running layer 3 mode. */
|
|
|
|
if (drm == DRM_LAYER3 &&
|
2016-12-13 09:40:41 -08:00
|
|
|
(interfaces->hw_address.htype != HTYPE_IPSEC_TUNNEL &&
|
|
|
|
interfaces->primary_address.s_addr == 0))
|
2017-02-13 11:15:39 -08:00
|
|
|
fatalx("interface '%s' does not have an address",
|
2016-12-08 01:29:50 -08:00
|
|
|
interfaces->name);
|
2004-04-12 14:10:28 -07:00
|
|
|
|
|
|
|
/* We need at least one server. */
|
2004-04-12 15:25:38 -07:00
|
|
|
if (!sp)
|
2004-04-12 14:31:14 -07:00
|
|
|
usage();
|
2004-04-12 14:10:28 -07:00
|
|
|
|
2009-11-03 02:14:09 -08:00
|
|
|
rdomain = get_rdomain(interfaces->name);
|
|
|
|
|
2009-09-03 04:56:49 -07:00
|
|
|
/* Enable the relay agent option by default for enc0 */
|
|
|
|
if (interfaces->hw_address.htype == HTYPE_IPSEC_TUNNEL)
|
|
|
|
oflag++;
|
|
|
|
|
2004-10-12 09:39:35 -07:00
|
|
|
bzero(&laddr, sizeof laddr);
|
|
|
|
laddr.sin_len = sizeof laddr;
|
|
|
|
laddr.sin_family = AF_INET;
|
2017-03-14 02:21:26 -07:00
|
|
|
laddr.sin_port = htons(SERVER_PORT);
|
2004-10-12 09:39:35 -07:00
|
|
|
laddr.sin_addr.s_addr = interfaces->primary_address.s_addr;
|
2004-04-12 14:10:28 -07:00
|
|
|
/* Set up the server sockaddrs. */
|
2004-04-12 15:25:38 -07:00
|
|
|
for (sp = servers; sp; sp = sp->next) {
|
2016-12-12 07:41:05 -08:00
|
|
|
if (sp->intf != NULL)
|
|
|
|
break;
|
|
|
|
|
2017-03-16 02:17:20 -07:00
|
|
|
ss2sin(&sp->to)->sin_port = htons(SERVER_PORT);
|
|
|
|
ss2sin(&sp->to)->sin_family = AF_INET;
|
|
|
|
ss2sin(&sp->to)->sin_len = sizeof(struct sockaddr_in);
|
2004-10-12 09:39:35 -07:00
|
|
|
sp->fd = socket(AF_INET, SOCK_DGRAM, 0);
|
|
|
|
if (sp->fd == -1)
|
2017-02-13 14:49:38 -08:00
|
|
|
fatal("socket");
|
2004-10-12 09:39:35 -07:00
|
|
|
opt = 1;
|
|
|
|
if (setsockopt(sp->fd, SOL_SOCKET, SO_REUSEPORT,
|
|
|
|
&opt, sizeof(opt)) == -1)
|
2017-02-13 14:49:38 -08:00
|
|
|
fatal("setsockopt");
|
2011-06-21 10:31:07 -07:00
|
|
|
if (setsockopt(sp->fd, SOL_SOCKET, SO_RTABLE, &rdomain,
|
2009-11-03 02:14:09 -08:00
|
|
|
sizeof(rdomain)) == -1)
|
2017-02-13 14:49:38 -08:00
|
|
|
fatal("setsockopt");
|
2017-02-13 14:05:35 -08:00
|
|
|
if (bind(sp->fd, (struct sockaddr *)&laddr, sizeof laddr) ==
|
|
|
|
-1)
|
2017-02-13 14:49:38 -08:00
|
|
|
fatal("bind");
|
2004-10-12 09:39:35 -07:00
|
|
|
if (connect(sp->fd, (struct sockaddr *)&sp->to,
|
2017-03-16 02:17:20 -07:00
|
|
|
sp->to.ss_len) == -1)
|
2017-02-13 14:49:38 -08:00
|
|
|
fatal("connect");
|
2004-10-12 09:39:35 -07:00
|
|
|
add_protocol("server", sp->fd, got_response, sp);
|
2004-04-12 14:10:28 -07:00
|
|
|
}
|
|
|
|
|
2009-11-03 02:14:09 -08:00
|
|
|
/* Socket used to forward packets to the DHCP client */
|
2009-09-03 04:56:49 -07:00
|
|
|
if (interfaces->hw_address.htype == HTYPE_IPSEC_TUNNEL) {
|
|
|
|
laddr.sin_addr.s_addr = INADDR_ANY;
|
|
|
|
server_fd = socket(AF_INET, SOCK_DGRAM, 0);
|
|
|
|
if (server_fd == -1)
|
2017-02-13 14:49:38 -08:00
|
|
|
fatal("socket");
|
2009-09-03 04:56:49 -07:00
|
|
|
opt = 1;
|
|
|
|
if (setsockopt(server_fd, SOL_SOCKET, SO_REUSEPORT,
|
|
|
|
&opt, sizeof(opt)) == -1)
|
2017-02-13 14:49:38 -08:00
|
|
|
fatal("setsockopt");
|
2011-06-21 10:31:07 -07:00
|
|
|
if (setsockopt(server_fd, SOL_SOCKET, SO_RTABLE, &rdomain,
|
2009-11-03 02:14:09 -08:00
|
|
|
sizeof(rdomain)) == -1)
|
2017-02-13 14:49:38 -08:00
|
|
|
fatal("setsockopt");
|
2009-09-03 04:56:49 -07:00
|
|
|
if (bind(server_fd, (struct sockaddr *)&laddr,
|
|
|
|
sizeof(laddr)) == -1)
|
2017-02-13 14:49:38 -08:00
|
|
|
fatal("bind");
|
2009-09-03 04:56:49 -07:00
|
|
|
}
|
|
|
|
|
2004-05-12 03:41:08 -07:00
|
|
|
tzset();
|
|
|
|
|
2004-04-12 14:10:28 -07:00
|
|
|
time(&cur_time);
|
2016-12-12 07:41:05 -08:00
|
|
|
if (drm == DRM_LAYER3)
|
|
|
|
bootp_packet_handler = relay;
|
|
|
|
else
|
|
|
|
bootp_packet_handler = l2relay;
|
2004-04-12 14:10:28 -07:00
|
|
|
|
2004-04-12 18:22:30 -07:00
|
|
|
if ((pw = getpwnam("_dhcp")) == NULL)
|
2017-02-13 11:15:39 -08:00
|
|
|
fatalx("user \"_dhcp\" not found");
|
2019-08-06 04:07:36 -07:00
|
|
|
if (chroot(pw->pw_dir) == -1)
|
2017-02-13 14:49:38 -08:00
|
|
|
fatal("chroot");
|
2004-04-12 18:22:30 -07:00
|
|
|
if (chdir("/") == -1)
|
2017-02-13 14:49:38 -08:00
|
|
|
fatal("chdir(\"/\")");
|
2004-04-12 18:22:30 -07:00
|
|
|
if (setgroups(1, &pw->pw_gid) ||
|
2005-05-01 19:29:26 -07:00
|
|
|
setresgid(pw->pw_gid, pw->pw_gid, pw->pw_gid) ||
|
|
|
|
setresuid(pw->pw_uid, pw->pw_uid, pw->pw_uid))
|
2017-02-13 14:49:38 -08:00
|
|
|
fatal("can't drop privileges");
|
2004-04-12 18:22:30 -07:00
|
|
|
|
2016-09-15 09:16:03 -07:00
|
|
|
if (daemonize) {
|
|
|
|
if (rdaemon(devnull) == -1)
|
2017-02-13 14:49:38 -08:00
|
|
|
fatal("rdaemon");
|
2017-07-05 04:11:56 -07:00
|
|
|
|
|
|
|
log_init(0, LOG_DAEMON); /* stop logging to stderr */
|
2016-09-15 09:16:03 -07:00
|
|
|
}
|
|
|
|
|
2016-12-07 08:41:17 -08:00
|
|
|
if (pledge("stdio route", NULL) == -1)
|
2017-02-13 11:15:39 -08:00
|
|
|
fatalx("pledge");
|
2016-12-07 08:41:17 -08:00
|
|
|
|
2004-04-12 15:25:38 -07:00
|
|
|
dispatch();
|
|
|
|
/* not reached */
|
2004-04-12 14:10:28 -07:00
|
|
|
|
2004-04-12 15:37:42 -07:00
|
|
|
exit(0);
|
2004-04-12 14:10:28 -07:00
|
|
|
}
|
|
|
|
|
2004-04-12 15:25:38 -07:00
|
|
|
void
|
|
|
|
relay(struct interface_info *ip, struct dhcp_packet *packet, int length,
|
2016-12-08 11:18:15 -08:00
|
|
|
struct packet_ctx *pc)
|
2004-04-12 14:10:28 -07:00
|
|
|
{
|
2004-04-12 15:25:38 -07:00
|
|
|
struct server_list *sp;
|
|
|
|
struct sockaddr_in to;
|
2004-04-12 14:10:28 -07:00
|
|
|
|
2004-04-12 15:25:38 -07:00
|
|
|
if (packet->hlen > sizeof packet->chaddr) {
|
2017-02-13 11:15:39 -08:00
|
|
|
log_info("Discarding packet with invalid hlen.");
|
2004-04-12 14:10:28 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If it's a bootreply, forward it to the client. */
|
2004-04-12 15:25:38 -07:00
|
|
|
if (packet->op == BOOTREPLY) {
|
2016-12-13 01:29:05 -08:00
|
|
|
/* Filter packet that were not meant for us. */
|
|
|
|
if (packet->giaddr.s_addr !=
|
|
|
|
interfaces->primary_address.s_addr)
|
|
|
|
return;
|
|
|
|
|
2004-04-12 15:25:38 -07:00
|
|
|
bzero(&to, sizeof(to));
|
|
|
|
if (!(packet->flags & htons(BOOTP_BROADCAST))) {
|
|
|
|
to.sin_addr = packet->yiaddr;
|
2017-03-14 02:21:26 -07:00
|
|
|
to.sin_port = htons(CLIENT_PORT);
|
2004-04-12 14:10:28 -07:00
|
|
|
} else {
|
2004-04-12 15:25:38 -07:00
|
|
|
to.sin_addr.s_addr = htonl(INADDR_BROADCAST);
|
2017-03-14 02:21:26 -07:00
|
|
|
to.sin_port = htons(CLIENT_PORT);
|
2004-04-12 14:10:28 -07:00
|
|
|
}
|
|
|
|
to.sin_family = AF_INET;
|
|
|
|
to.sin_len = sizeof to;
|
2016-12-08 11:18:15 -08:00
|
|
|
*ss2sin(&pc->pc_dst) = to;
|
2004-04-12 14:10:28 -07:00
|
|
|
|
2016-12-07 11:51:48 -08:00
|
|
|
/*
|
|
|
|
* Set up the hardware destination address. If it's a reply
|
|
|
|
* with the BROADCAST flag set, we should send an L2 broad-
|
|
|
|
* cast as well.
|
|
|
|
*/
|
|
|
|
if (!(packet->flags & htons(BOOTP_BROADCAST))) {
|
2016-12-08 11:18:15 -08:00
|
|
|
pc->pc_hlen = packet->hlen;
|
|
|
|
if (pc->pc_hlen > CHADDR_SIZE)
|
|
|
|
pc->pc_hlen = CHADDR_SIZE;
|
|
|
|
memcpy(pc->pc_dmac, packet->chaddr, pc->pc_hlen);
|
|
|
|
pc->pc_htype = packet->htype;
|
2016-12-07 11:51:48 -08:00
|
|
|
} else {
|
2016-12-08 11:18:15 -08:00
|
|
|
memset(pc->pc_dmac, 0xff, sizeof(pc->pc_dmac));
|
2016-12-07 11:51:48 -08:00
|
|
|
}
|
2004-04-12 14:10:28 -07:00
|
|
|
|
2016-12-13 07:28:19 -08:00
|
|
|
relay_agentinfo(pc, interfaces, packet->op);
|
2016-12-12 07:41:05 -08:00
|
|
|
if ((length = relay_agentinfo_remove(pc, packet,
|
|
|
|
length)) == -1) {
|
2017-02-13 11:15:39 -08:00
|
|
|
log_info("ignoring BOOTREPLY with invalid "
|
2009-09-03 04:56:49 -07:00
|
|
|
"relay agent information");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-01-15 15:45:03 -08:00
|
|
|
/*
|
|
|
|
* VMware PXE "ROMs" confuse the DHCP gateway address
|
|
|
|
* with the IP gateway address. This is a problem if your
|
|
|
|
* DHCP relay is running on something that's not your
|
|
|
|
* network gateway.
|
|
|
|
*
|
|
|
|
* It is purely informational from the relay to the client
|
|
|
|
* so we can safely clear it.
|
|
|
|
*/
|
|
|
|
packet->giaddr.s_addr = 0x0;
|
|
|
|
|
2016-12-08 11:18:15 -08:00
|
|
|
ss2sin(&pc->pc_src)->sin_addr = interfaces->primary_address;
|
|
|
|
if (send_packet(interfaces, packet, length, pc) != -1)
|
2017-02-13 11:15:39 -08:00
|
|
|
log_debug("forwarded BOOTREPLY for %s to %s",
|
2004-04-12 15:25:38 -07:00
|
|
|
print_hw_addr(packet->htype, packet->hlen,
|
2004-04-12 16:09:32 -07:00
|
|
|
packet->chaddr), inet_ntoa(to.sin_addr));
|
2004-04-12 14:10:28 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2004-10-12 09:39:35 -07:00
|
|
|
if (ip == NULL) {
|
2017-02-13 11:15:39 -08:00
|
|
|
log_info("ignoring non BOOTREPLY from server");
|
2004-10-12 09:39:35 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-12-07 12:03:22 -08:00
|
|
|
if (packet->hops > 16) {
|
2017-02-13 11:15:39 -08:00
|
|
|
log_info("ignoring BOOTREQUEST with hop count of %d",
|
2016-12-07 12:03:22 -08:00
|
|
|
packet->hops);
|
2004-04-12 14:10:28 -07:00
|
|
|
return;
|
|
|
|
}
|
2016-12-07 12:03:22 -08:00
|
|
|
packet->hops++;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set the giaddr so the server can figure out what net it's
|
|
|
|
* from and so that we can later forward the response to the
|
|
|
|
* correct net. The RFC specifies that we have to keep the
|
|
|
|
* initial giaddr (in case we relay over multiple hops).
|
|
|
|
*/
|
|
|
|
if (!packet->giaddr.s_addr)
|
|
|
|
packet->giaddr = ip->primary_address;
|
2004-04-12 14:10:28 -07:00
|
|
|
|
2016-12-13 07:28:19 -08:00
|
|
|
relay_agentinfo(pc, interfaces, packet->op);
|
2016-12-12 07:41:05 -08:00
|
|
|
if ((length = relay_agentinfo_append(pc, packet, length)) == -1) {
|
2017-02-13 11:15:39 -08:00
|
|
|
log_info("ignoring BOOTREQUEST with invalid "
|
2009-09-03 04:56:49 -07:00
|
|
|
"relay agent information");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2004-04-12 14:10:28 -07:00
|
|
|
/* Otherwise, it's a BOOTREQUEST, so forward it to all the
|
|
|
|
servers. */
|
2004-04-12 15:25:38 -07:00
|
|
|
for (sp = servers; sp; sp = sp->next) {
|
2004-10-12 09:39:35 -07:00
|
|
|
if (send(sp->fd, packet, length, 0) != -1) {
|
2017-02-13 11:15:39 -08:00
|
|
|
log_debug("forwarded BOOTREQUEST for %s to %s",
|
2004-04-12 15:25:38 -07:00
|
|
|
print_hw_addr(packet->htype, packet->hlen,
|
2017-03-16 02:17:20 -07:00
|
|
|
packet->chaddr),
|
|
|
|
inet_ntoa(ss2sin(&sp->to)->sin_addr));
|
2004-04-12 14:10:28 -07:00
|
|
|
}
|
|
|
|
}
|
2004-04-12 15:25:38 -07:00
|
|
|
|
2004-04-12 14:10:28 -07:00
|
|
|
}
|
|
|
|
|
2004-04-12 14:31:14 -07:00
|
|
|
void
|
|
|
|
usage(void)
|
2004-04-12 14:10:28 -07:00
|
|
|
{
|
2004-04-12 14:31:14 -07:00
|
|
|
extern char *__progname;
|
|
|
|
|
2016-12-16 10:38:39 -08:00
|
|
|
fprintf(stderr, "usage: %s [-dor] [-C circuit-id] [-R remote-id] "
|
2016-12-12 22:55:32 -08:00
|
|
|
"-i interface\n\tdestination ...\n",
|
2008-07-09 12:41:56 -07:00
|
|
|
__progname);
|
2004-04-12 15:37:42 -07:00
|
|
|
exit(1);
|
2004-04-12 14:10:28 -07:00
|
|
|
}
|
|
|
|
|
2016-09-15 09:16:03 -07:00
|
|
|
int
|
|
|
|
rdaemon(int devnull)
|
|
|
|
{
|
2016-09-26 10:15:19 -07:00
|
|
|
if (devnull == -1) {
|
|
|
|
errno = EBADF;
|
|
|
|
return (-1);
|
|
|
|
}
|
|
|
|
if (fcntl(devnull, F_GETFL) == -1)
|
|
|
|
return (-1);
|
2016-09-15 09:16:03 -07:00
|
|
|
|
|
|
|
switch (fork()) {
|
|
|
|
case -1:
|
|
|
|
return (-1);
|
|
|
|
case 0:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
_exit(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (setsid() == -1)
|
|
|
|
return (-1);
|
|
|
|
|
|
|
|
(void)dup2(devnull, STDIN_FILENO);
|
|
|
|
(void)dup2(devnull, STDOUT_FILENO);
|
|
|
|
(void)dup2(devnull, STDERR_FILENO);
|
|
|
|
if (devnull > 2)
|
|
|
|
(void)close(devnull);
|
|
|
|
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
2004-04-12 15:25:38 -07:00
|
|
|
char *
|
|
|
|
print_hw_addr(int htype, int hlen, unsigned char *data)
|
2004-04-12 14:10:28 -07:00
|
|
|
{
|
2004-04-12 15:25:38 -07:00
|
|
|
static char habuf[49];
|
2005-11-13 12:25:59 -08:00
|
|
|
char *s = habuf;
|
|
|
|
int i, j, slen = sizeof(habuf);
|
|
|
|
|
|
|
|
if (htype == 0 || hlen == 0) {
|
2004-04-12 15:25:38 -07:00
|
|
|
bad:
|
2005-11-13 12:25:59 -08:00
|
|
|
strlcpy(habuf, "<null>", sizeof habuf);
|
|
|
|
return habuf;
|
|
|
|
}
|
2004-04-12 14:10:28 -07:00
|
|
|
|
2005-11-13 12:25:59 -08:00
|
|
|
for (i = 0; i < hlen; i++) {
|
|
|
|
j = snprintf(s, slen, "%02x", data[i]);
|
|
|
|
if (j <= 0 || j >= slen)
|
|
|
|
goto bad;
|
|
|
|
j = strlen (s);
|
|
|
|
s += j;
|
|
|
|
slen -= (j + 1);
|
|
|
|
*s++ = ':';
|
|
|
|
}
|
|
|
|
*--s = '\0';
|
|
|
|
return habuf;
|
2004-04-12 14:10:28 -07:00
|
|
|
}
|
2004-10-12 09:39:35 -07:00
|
|
|
|
|
|
|
void
|
|
|
|
got_response(struct protocol *l)
|
|
|
|
{
|
2016-12-08 11:18:15 -08:00
|
|
|
struct packet_ctx pc;
|
2004-10-12 09:39:35 -07:00
|
|
|
ssize_t result;
|
|
|
|
union {
|
|
|
|
/*
|
|
|
|
* Packet input buffer. Must be as large as largest
|
|
|
|
* possible MTU.
|
|
|
|
*/
|
|
|
|
unsigned char packbuf[4095];
|
|
|
|
struct dhcp_packet packet;
|
|
|
|
} u;
|
|
|
|
struct server_list *sp = l->local;
|
|
|
|
|
2009-09-03 04:56:49 -07:00
|
|
|
memset(&u, DHO_END, sizeof(u));
|
2004-10-12 09:39:35 -07:00
|
|
|
if ((result = recv(l->fd, u.packbuf, sizeof(u), 0)) == -1 &&
|
|
|
|
errno != ECONNREFUSED) {
|
|
|
|
/*
|
2013-03-03 16:29:56 -08:00
|
|
|
* Ignore ECONNREFUSED as too many dhcp servers send a bogus
|
2004-10-12 09:39:35 -07:00
|
|
|
* icmp unreach for every request.
|
|
|
|
*/
|
2017-03-16 02:17:20 -07:00
|
|
|
log_warn("recv failed for %s",
|
|
|
|
inet_ntoa(ss2sin(&sp->to)->sin_addr));
|
2004-10-12 09:39:35 -07:00
|
|
|
return;
|
|
|
|
}
|
2007-02-09 08:49:47 -08:00
|
|
|
if (result == -1 && errno == ECONNREFUSED)
|
|
|
|
return;
|
|
|
|
|
2004-10-12 09:39:35 -07:00
|
|
|
if (result == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (result < BOOTP_MIN_LEN) {
|
2017-02-13 11:15:39 -08:00
|
|
|
log_info("Discarding packet with invalid size.");
|
2004-10-12 09:39:35 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-12-08 11:18:15 -08:00
|
|
|
memset(&pc, 0, sizeof(pc));
|
|
|
|
pc.pc_src.ss_family = AF_INET;
|
|
|
|
pc.pc_src.ss_len = sizeof(struct sockaddr_in);
|
2017-03-16 02:17:20 -07:00
|
|
|
memcpy(&ss2sin(&pc.pc_src)->sin_addr, &ss2sin(&sp->to)->sin_addr,
|
2016-12-08 11:18:15 -08:00
|
|
|
sizeof(ss2sin(&pc.pc_src)->sin_addr));
|
2017-03-14 02:21:26 -07:00
|
|
|
ss2sin(&pc.pc_src)->sin_port = htons(SERVER_PORT);
|
2004-10-12 09:39:35 -07:00
|
|
|
|
2016-12-08 11:18:15 -08:00
|
|
|
pc.pc_dst.ss_family = AF_INET;
|
|
|
|
pc.pc_dst.ss_len = sizeof(struct sockaddr_in);
|
2017-03-14 02:21:26 -07:00
|
|
|
ss2sin(&pc.pc_dst)->sin_port = htons(CLIENT_PORT);
|
2016-12-08 11:18:15 -08:00
|
|
|
|
|
|
|
if (bootp_packet_handler)
|
|
|
|
(*bootp_packet_handler)(NULL, &u.packet, result, &pc);
|
2004-10-12 09:39:35 -07:00
|
|
|
}
|
2009-09-03 04:56:49 -07:00
|
|
|
|
2016-12-12 07:41:05 -08:00
|
|
|
void
|
2016-12-13 07:28:19 -08:00
|
|
|
relay_agentinfo(struct packet_ctx *pc, struct interface_info *intf,
|
|
|
|
int bootop)
|
2016-12-12 07:41:05 -08:00
|
|
|
{
|
2016-12-13 07:28:19 -08:00
|
|
|
static u_int8_t buf[8];
|
|
|
|
struct sockaddr_in *sin;
|
2016-12-12 07:41:05 -08:00
|
|
|
|
|
|
|
if (oflag == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (rai_remote != NULL) {
|
2017-04-04 08:50:29 -07:00
|
|
|
pc->pc_remote = rai_remote;
|
2016-12-12 07:41:05 -08:00
|
|
|
pc->pc_remotelen = strlen(rai_remote);
|
|
|
|
} else
|
|
|
|
pc->pc_remotelen = 0;
|
|
|
|
|
|
|
|
if (rai_circuit == NULL) {
|
|
|
|
buf[0] = (uint8_t)(intf->index << 8);
|
|
|
|
buf[1] = intf->index & 0xff;
|
|
|
|
pc->pc_circuit = buf;
|
|
|
|
pc->pc_circuitlen = 2;
|
|
|
|
|
|
|
|
if (rai_remote == NULL) {
|
2016-12-13 07:28:19 -08:00
|
|
|
if (bootop == BOOTREPLY)
|
|
|
|
sin = ss2sin(&pc->pc_dst);
|
|
|
|
else
|
|
|
|
sin = ss2sin(&pc->pc_src);
|
|
|
|
|
2016-12-12 07:41:05 -08:00
|
|
|
pc->pc_remote =
|
2016-12-13 07:28:19 -08:00
|
|
|
(uint8_t *)&sin->sin_addr;
|
2016-12-12 07:41:05 -08:00
|
|
|
pc->pc_remotelen =
|
2016-12-13 07:28:19 -08:00
|
|
|
sizeof(sin->sin_addr);
|
2016-12-12 07:41:05 -08:00
|
|
|
}
|
|
|
|
} else {
|
2017-04-04 08:50:29 -07:00
|
|
|
pc->pc_circuit = rai_circuit;
|
2016-12-12 07:41:05 -08:00
|
|
|
pc->pc_circuitlen = strlen(rai_circuit);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
relay_agentinfo_cmp(struct packet_ctx *pc, uint8_t *p, int plen)
|
|
|
|
{
|
|
|
|
int len;
|
|
|
|
char buf[256];
|
|
|
|
|
|
|
|
if (oflag == 0)
|
|
|
|
return (-1);
|
|
|
|
|
|
|
|
len = *(p + 1);
|
|
|
|
if (len > plen)
|
|
|
|
return (-1);
|
|
|
|
|
|
|
|
switch (*p) {
|
|
|
|
case RAI_CIRCUIT_ID:
|
|
|
|
if (pc->pc_circuit == NULL)
|
|
|
|
return (-1);
|
|
|
|
if (pc->pc_circuitlen != len)
|
|
|
|
return (-1);
|
|
|
|
|
|
|
|
memcpy(buf, p + DHCP_OPTION_HDR_LEN, len);
|
|
|
|
return (memcmp(pc->pc_circuit, buf, len));
|
|
|
|
|
|
|
|
case RAI_REMOTE_ID:
|
|
|
|
if (pc->pc_remote == NULL)
|
|
|
|
return (-1);
|
|
|
|
if (pc->pc_remotelen != len)
|
|
|
|
return (-1);
|
|
|
|
|
|
|
|
memcpy(buf, p + DHCP_OPTION_HDR_LEN, len);
|
|
|
|
return (memcmp(pc->pc_remote, buf, len));
|
|
|
|
|
|
|
|
default:
|
|
|
|
/* Unmatched type */
|
2017-02-13 11:15:39 -08:00
|
|
|
log_info("unmatched relay info %d", *p);
|
2016-12-12 07:41:05 -08:00
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-03 04:56:49 -07:00
|
|
|
ssize_t
|
2016-12-12 07:41:05 -08:00
|
|
|
relay_agentinfo_append(struct packet_ctx *pc, struct dhcp_packet *dp,
|
|
|
|
size_t dplen)
|
2009-09-03 04:56:49 -07:00
|
|
|
{
|
2016-12-12 07:41:05 -08:00
|
|
|
uint8_t *p, *startp;
|
|
|
|
ssize_t newtotal = dplen;
|
|
|
|
int opttotal, optlen, i, hasinfo = 0;
|
|
|
|
int maxlen, neededlen;
|
|
|
|
|
|
|
|
/* Only append when enabled. */
|
|
|
|
if (oflag == 0)
|
|
|
|
return (dplen);
|
|
|
|
|
|
|
|
startp = (uint8_t *)dp;
|
|
|
|
p = (uint8_t *)&dp->options;
|
|
|
|
if (memcmp(p, DHCP_OPTIONS_COOKIE, DHCP_OPTIONS_COOKIE_LEN)) {
|
2017-02-13 11:15:39 -08:00
|
|
|
log_info("invalid dhcp options cookie");
|
2016-12-12 07:41:05 -08:00
|
|
|
return (-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
p += DHCP_OPTIONS_COOKIE_LEN;
|
|
|
|
opttotal = dplen - DHCP_FIXED_NON_UDP - DHCP_OPTIONS_COOKIE_LEN;
|
|
|
|
maxlen = DHCP_MTU_MAX - DHCP_FIXED_LEN - DHCP_OPTIONS_COOKIE_LEN - 1;
|
|
|
|
if (maxlen < 1 || opttotal < 1)
|
|
|
|
return (dplen);
|
|
|
|
|
|
|
|
for (i = 0; i < opttotal && *p != DHO_END;) {
|
|
|
|
if (*p == DHO_PAD)
|
|
|
|
optlen = 1;
|
|
|
|
else
|
|
|
|
optlen = p[1] + DHCP_OPTION_HDR_LEN;
|
|
|
|
|
|
|
|
if ((i + optlen) > opttotal) {
|
2017-02-13 11:15:39 -08:00
|
|
|
log_info("truncated dhcp options");
|
2016-12-12 07:41:05 -08:00
|
|
|
return (-1);
|
|
|
|
}
|
2009-09-03 04:56:49 -07:00
|
|
|
|
2016-12-12 07:41:05 -08:00
|
|
|
if (*p == DHO_RELAY_AGENT_INFORMATION) {
|
2016-12-16 10:38:39 -08:00
|
|
|
if (rai_replace) {
|
|
|
|
memmove(p, p + optlen, opttotal - i);
|
|
|
|
opttotal -= optlen;
|
|
|
|
optlen = 0;
|
|
|
|
} else
|
|
|
|
hasinfo = 1;
|
2016-12-12 07:41:05 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
p += optlen;
|
|
|
|
i += optlen;
|
|
|
|
|
|
|
|
/* We reached the end, append the relay agent info. */
|
|
|
|
if (i < opttotal && *p == DHO_END) {
|
|
|
|
/* We already have the Relay Agent Info, skip it. */
|
|
|
|
if (hasinfo)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* Calculate needed length to append new data. */
|
|
|
|
neededlen = newtotal + DHCP_OPTION_HDR_LEN;
|
|
|
|
if (pc->pc_circuitlen > 0)
|
|
|
|
neededlen += DHCP_OPTION_HDR_LEN +
|
|
|
|
pc->pc_circuitlen;
|
|
|
|
if (pc->pc_remotelen > 0)
|
|
|
|
neededlen += DHCP_OPTION_HDR_LEN +
|
|
|
|
pc->pc_remotelen;
|
|
|
|
|
|
|
|
/* Save one byte for DHO_END. */
|
|
|
|
neededlen += 1;
|
|
|
|
|
2017-02-13 14:05:35 -08:00
|
|
|
/* Check if we have space for the new options. */
|
2016-12-12 07:41:05 -08:00
|
|
|
if (neededlen > maxlen) {
|
2017-02-13 11:15:39 -08:00
|
|
|
log_warnx("no space for relay agent info");
|
2016-12-12 07:41:05 -08:00
|
|
|
return (newtotal);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* New option header: 2 bytes. */
|
|
|
|
newtotal += DHCP_OPTION_HDR_LEN;
|
|
|
|
|
|
|
|
*p++ = DHO_RELAY_AGENT_INFORMATION;
|
|
|
|
*p = 0;
|
|
|
|
if (pc->pc_circuitlen > 0) {
|
|
|
|
newtotal += DHCP_OPTION_HDR_LEN +
|
|
|
|
pc->pc_circuitlen;
|
|
|
|
*p = (*p) + DHCP_OPTION_HDR_LEN +
|
|
|
|
pc->pc_circuitlen;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pc->pc_remotelen > 0) {
|
|
|
|
newtotal += DHCP_OPTION_HDR_LEN +
|
|
|
|
pc->pc_remotelen;
|
|
|
|
*p = (*p) + DHCP_OPTION_HDR_LEN +
|
|
|
|
pc->pc_remotelen;
|
|
|
|
}
|
|
|
|
|
|
|
|
p++;
|
|
|
|
|
|
|
|
/* Sub-option circuit-id header plus value. */
|
|
|
|
if (pc->pc_circuitlen > 0) {
|
|
|
|
*p++ = RAI_CIRCUIT_ID;
|
|
|
|
*p++ = pc->pc_circuitlen;
|
|
|
|
memcpy(p, pc->pc_circuit, pc->pc_circuitlen);
|
|
|
|
|
|
|
|
p += pc->pc_circuitlen;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Sub-option remote-id header plus value. */
|
|
|
|
if (pc->pc_remotelen > 0) {
|
|
|
|
*p++ = RAI_REMOTE_ID;
|
|
|
|
*p++ = pc->pc_remotelen;
|
|
|
|
memcpy(p, pc->pc_remote, pc->pc_remotelen);
|
|
|
|
|
|
|
|
p += pc->pc_remotelen;
|
|
|
|
}
|
|
|
|
|
|
|
|
*p = DHO_END;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Zero the padding so we don't leak anything. */
|
|
|
|
p++;
|
|
|
|
if (p < (startp + maxlen))
|
|
|
|
memset(p, 0, (startp + maxlen) - p);
|
|
|
|
|
|
|
|
return (newtotal);
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t
|
|
|
|
relay_agentinfo_remove(struct packet_ctx *pc, struct dhcp_packet *dp,
|
|
|
|
size_t dplen)
|
|
|
|
{
|
|
|
|
uint8_t *p, *np, *startp, *endp;
|
|
|
|
int opttotal, optleft;
|
|
|
|
int suboptlen, optlen, i;
|
|
|
|
int maxlen, remaining, matched = 0;
|
|
|
|
|
|
|
|
startp = (uint8_t *)dp;
|
|
|
|
p = (uint8_t *)&dp->options;
|
|
|
|
if (memcmp(p, DHCP_OPTIONS_COOKIE, DHCP_OPTIONS_COOKIE_LEN)) {
|
2017-02-13 11:15:39 -08:00
|
|
|
log_info("invalid dhcp options cookie");
|
2016-12-12 07:41:05 -08:00
|
|
|
return (-1);
|
|
|
|
}
|
2009-09-03 04:56:49 -07:00
|
|
|
|
|
|
|
maxlen = DHCP_MTU_MAX - DHCP_FIXED_LEN - DHCP_OPTIONS_COOKIE_LEN - 1;
|
2016-12-12 07:41:05 -08:00
|
|
|
opttotal = dplen - DHCP_FIXED_NON_UDP - DHCP_OPTIONS_COOKIE_LEN;
|
|
|
|
optleft = opttotal;
|
2009-09-03 04:56:49 -07:00
|
|
|
|
2016-12-12 07:41:05 -08:00
|
|
|
p += DHCP_OPTIONS_COOKIE_LEN;
|
|
|
|
endp = p + opttotal;
|
2009-09-03 04:56:49 -07:00
|
|
|
|
2016-12-12 07:41:05 -08:00
|
|
|
for (i = 0; i < opttotal && *p != DHO_END;) {
|
2009-09-03 04:56:49 -07:00
|
|
|
if (*p == DHO_PAD)
|
2016-12-12 07:41:05 -08:00
|
|
|
optlen = 1;
|
2009-09-03 04:56:49 -07:00
|
|
|
else
|
2016-12-12 07:41:05 -08:00
|
|
|
optlen = p[1] + DHCP_OPTION_HDR_LEN;
|
2009-09-03 04:56:49 -07:00
|
|
|
|
2016-12-12 07:41:05 -08:00
|
|
|
if ((i + optlen) > opttotal) {
|
2017-02-13 11:15:39 -08:00
|
|
|
log_info("truncated dhcp options");
|
2016-12-12 07:41:05 -08:00
|
|
|
return (-1);
|
2009-09-03 04:56:49 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (*p == DHO_RELAY_AGENT_INFORMATION) {
|
2016-12-12 07:41:05 -08:00
|
|
|
/* Fast case: there is no next option. */
|
|
|
|
np = p + optlen;
|
|
|
|
if (*np == DHO_END) {
|
|
|
|
*p = *np;
|
|
|
|
endp = p + 1;
|
|
|
|
/* Zero the padding so we don't leak data. */
|
|
|
|
if (endp < (startp + maxlen))
|
|
|
|
memset(endp, 0,
|
|
|
|
(startp + maxlen) - endp);
|
|
|
|
|
|
|
|
return (dplen);
|
2009-09-03 04:56:49 -07:00
|
|
|
}
|
|
|
|
|
2016-12-12 07:41:05 -08:00
|
|
|
remaining = optlen;
|
|
|
|
while (remaining > 0) {
|
|
|
|
suboptlen = *(p + 1);
|
|
|
|
remaining -= DHCP_OPTION_HDR_LEN + suboptlen;
|
2009-09-03 04:56:49 -07:00
|
|
|
|
2016-12-12 07:41:05 -08:00
|
|
|
matched = 1;
|
|
|
|
if (relay_agentinfo_cmp(pc, p, suboptlen) == 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
matched = 0;
|
2009-09-03 04:56:49 -07:00
|
|
|
break;
|
|
|
|
}
|
2016-12-12 07:41:05 -08:00
|
|
|
/* It is not ours Relay Agent Info, don't remove it. */
|
|
|
|
if (matched == 0)
|
|
|
|
break;
|
2009-09-03 04:56:49 -07:00
|
|
|
|
2016-12-12 07:41:05 -08:00
|
|
|
/* Move the other options on top of this one. */
|
|
|
|
optleft -= optlen;
|
|
|
|
endp -= optlen;
|
2009-09-03 04:56:49 -07:00
|
|
|
|
2016-12-12 07:41:05 -08:00
|
|
|
/* Replace the old agent relay info. */
|
|
|
|
memmove(p, dp, optleft);
|
|
|
|
|
|
|
|
endp++;
|
|
|
|
/* Zero the padding so we don't leak data. */
|
|
|
|
if (endp < (startp + maxlen))
|
|
|
|
memset(endp, 0,
|
|
|
|
(startp + maxlen) - endp);
|
|
|
|
|
|
|
|
return (endp - startp);
|
2009-09-03 04:56:49 -07:00
|
|
|
}
|
2016-12-12 07:41:05 -08:00
|
|
|
|
|
|
|
p += optlen;
|
|
|
|
i += optlen;
|
|
|
|
optleft -= optlen;
|
2009-09-03 04:56:49 -07:00
|
|
|
}
|
|
|
|
|
2016-12-12 07:41:05 -08:00
|
|
|
return (endp - startp);
|
2009-09-03 04:56:49 -07:00
|
|
|
}
|
2009-11-03 02:14:09 -08:00
|
|
|
|
|
|
|
int
|
|
|
|
get_rdomain(char *name)
|
|
|
|
{
|
|
|
|
int rv = 0, s;
|
|
|
|
struct ifreq ifr;
|
|
|
|
|
|
|
|
if ((s = socket(AF_INET, SOCK_DGRAM, 0)) == -1)
|
2017-02-13 14:49:38 -08:00
|
|
|
fatal("get_rdomain socket");
|
2009-11-03 02:14:09 -08:00
|
|
|
|
|
|
|
bzero(&ifr, sizeof(ifr));
|
|
|
|
strlcpy(ifr.ifr_name, name, sizeof(ifr.ifr_name));
|
2010-07-02 21:44:50 -07:00
|
|
|
if (ioctl(s, SIOCGIFRDOMAIN, (caddr_t)&ifr) != -1)
|
2009-11-03 02:14:09 -08:00
|
|
|
rv = ifr.ifr_rdomainid;
|
|
|
|
|
|
|
|
close(s);
|
|
|
|
return rv;
|
|
|
|
}
|
2016-12-12 07:41:05 -08:00
|
|
|
|
|
|
|
void
|
|
|
|
l2relay(struct interface_info *ip, struct dhcp_packet *dp, int length,
|
|
|
|
struct packet_ctx *pc)
|
|
|
|
{
|
|
|
|
struct server_list *sp;
|
|
|
|
ssize_t dplen;
|
|
|
|
|
|
|
|
if (dp->hlen > sizeof(dp->chaddr)) {
|
2017-02-13 11:15:39 -08:00
|
|
|
log_info("Discarding packet with invalid hlen.");
|
2016-12-12 07:41:05 -08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-12-13 07:28:19 -08:00
|
|
|
relay_agentinfo(pc, ip, dp->op);
|
2016-12-12 07:41:05 -08:00
|
|
|
|
|
|
|
switch (dp->op) {
|
|
|
|
case BOOTREQUEST:
|
|
|
|
/* Add the relay agent info asked by the user. */
|
|
|
|
if ((dplen = relay_agentinfo_append(pc, dp, length)) == -1)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Re-send the packet to every interface except the one
|
|
|
|
* it came in.
|
|
|
|
*/
|
|
|
|
for (sp = servers; sp != NULL; sp = sp->next) {
|
|
|
|
if (sp->intf == ip)
|
|
|
|
continue;
|
|
|
|
|
2017-02-13 11:15:39 -08:00
|
|
|
log_debug("forwarded BOOTREQUEST for %s to %s",
|
2016-12-12 07:41:05 -08:00
|
|
|
print_hw_addr(pc->pc_htype, pc->pc_hlen,
|
|
|
|
pc->pc_smac), sp->intf->name);
|
|
|
|
|
|
|
|
send_packet(sp->intf, dp, dplen, pc);
|
|
|
|
}
|
|
|
|
if (ip != interfaces) {
|
2017-02-13 11:15:39 -08:00
|
|
|
log_debug("forwarded BOOTREQUEST for %s to %s",
|
2016-12-12 07:41:05 -08:00
|
|
|
print_hw_addr(pc->pc_htype, pc->pc_hlen,
|
|
|
|
pc->pc_smac), interfaces->name);
|
|
|
|
|
|
|
|
send_packet(interfaces, dp, dplen, pc);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BOOTREPLY:
|
|
|
|
/* Remove relay agent info on offer. */
|
|
|
|
if ((dplen = relay_agentinfo_remove(pc, dp, length)) == -1)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (ip != interfaces) {
|
2017-02-13 11:15:39 -08:00
|
|
|
log_debug("forwarded BOOTREPLY for %s to %s",
|
2016-12-12 07:41:05 -08:00
|
|
|
print_hw_addr(pc->pc_htype, pc->pc_hlen,
|
|
|
|
pc->pc_dmac), interfaces->name);
|
|
|
|
send_packet(interfaces, dp, dplen, pc);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2017-02-13 11:15:39 -08:00
|
|
|
log_debug("invalid operation type '%d'", dp->op);
|
2016-12-12 07:41:05 -08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|