summaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
authorSteven Barth <steven@midlink.org>2014-01-04 10:14:59 +0100
committerSteven Barth <steven@midlink.org>2014-01-04 10:14:59 +0100
commit6deafcc1fad03312b4df0778ef8ebef11fb93e34 (patch)
tree633c3e0e90fa948cd5624964ff88a96cd090995b
parent3fcc0814db3773e1295a372ea7a4194e19c62e52 (diff)
Use 1 socket per interface for DHCPv6
-rw-r--r--src/dhcpv6-ia.c6
-rw-r--r--src/dhcpv6.c92
-rw-r--r--src/dhcpv6.h2
-rw-r--r--src/odhcpd.h1
4 files changed, 50 insertions, 51 deletions
diff --git a/src/dhcpv6-ia.c b/src/dhcpv6-ia.c
index e5a7c0b..f933da6 100644
--- a/src/dhcpv6-ia.c
+++ b/src/dhcpv6-ia.c
@@ -32,13 +32,11 @@
static void update(struct interface *iface);
static void reconf_timer(struct uloop_timeout *event);
static struct uloop_timeout reconf_event = {.cb = reconf_timer};
-static int socket_fd = -1;
static uint32_t serial = 0;
-int dhcpv6_ia_init(int dhcpv6_socket)
+int dhcpv6_ia_init(void)
{
- socket_fd = dhcpv6_socket;
uloop_timeout_set(&reconf_event, 2000);
return 0;
}
@@ -183,7 +181,7 @@ static int send_reconf(struct interface *iface, struct dhcpv6_assignment *assign
md5_hash(reconf_msg.auth.key, 16, &md5);
md5_end(reconf_msg.auth.key, &md5);
- return odhcpd_send(socket_fd, &assign->peer, &iov, 1, iface);
+ return odhcpd_send(iface->dhcpv6_event.uloop.fd, &assign->peer, &iov, 1, iface);
}
diff --git a/src/dhcpv6.c b/src/dhcpv6.c
index 7560a75..8f8f435 100644
--- a/src/dhcpv6.c
+++ b/src/dhcpv6.c
@@ -32,68 +32,68 @@ static void handle_dhcpv6(void *addr, void *data, size_t len,
static void handle_client_request(void *addr, void *data, size_t len,
struct interface *iface);
-static struct odhcpd_event dhcpv6_event = {{.fd = -1}, handle_dhcpv6};
-
// Create socket and register events
int init_dhcpv6(void)
{
- int sock = socket(AF_INET6, SOCK_DGRAM | SOCK_CLOEXEC, IPPROTO_UDP);
- if (sock < 0) {
- syslog(LOG_ERR, "Failed to create DHCPv6 server socket: %s",
- strerror(errno));
- return -1;
- }
-
- // Basic IPv6 configuration
- int val = 1;
- setsockopt(sock, IPPROTO_IPV6, IPV6_V6ONLY, &val, sizeof(val));
- setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, &val, sizeof(val));
- setsockopt(sock, IPPROTO_IPV6, IPV6_RECVPKTINFO, &val, sizeof(val));
-
- val = DHCPV6_HOP_COUNT_LIMIT;
- setsockopt(sock, IPPROTO_IPV6, IPV6_MULTICAST_HOPS, &val, sizeof(val));
-
- val = 0;
- setsockopt(sock, IPPROTO_IPV6, IPV6_MULTICAST_LOOP, &val, sizeof(val));
+ dhcpv6_ia_init();
+ return 0;
+}
- struct sockaddr_in6 bind_addr = {AF_INET6, htons(DHCPV6_SERVER_PORT),
- 0, IN6ADDR_ANY_INIT, 0};
- if (bind(sock, (struct sockaddr*)&bind_addr, sizeof(bind_addr))) {
- syslog(LOG_ERR, "Failed to open DHCPv6 server socket: %s",
- strerror(errno));
- return -1;
+int setup_dhcpv6_interface(struct interface *iface, bool enable)
+{
+ if (iface->dhcpv6_event.uloop.fd > 0) {
+ uloop_fd_delete(&iface->dhcpv6_event.uloop);
+ close(iface->dhcpv6_event.uloop.fd);
+ iface->dhcpv6_event.uloop.fd = -1;
}
- dhcpv6_event.uloop.fd = sock;
- odhcpd_register(&dhcpv6_event);
+ // Configure multicast settings
+ if (enable && iface->dhcpv6 && !iface->master) {
+ int sock = socket(AF_INET6, SOCK_DGRAM | SOCK_CLOEXEC, IPPROTO_UDP);
+ if (sock < 0) {
+ syslog(LOG_ERR, "Failed to create DHCPv6 server socket: %s",
+ strerror(errno));
+ return -1;
+ }
- dhcpv6_ia_init(dhcpv6_event.uloop.fd);
+ // Basic IPv6 configuration
+ setsockopt(sock, SOL_SOCKET, SO_BINDTODEVICE, iface->ifname, strlen(iface->ifname));
- return 0;
-}
+ int val = 1;
+ setsockopt(sock, IPPROTO_IPV6, IPV6_V6ONLY, &val, sizeof(val));
+ setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, &val, sizeof(val));
+ setsockopt(sock, IPPROTO_IPV6, IPV6_RECVPKTINFO, &val, sizeof(val));
+ val = DHCPV6_HOP_COUNT_LIMIT;
+ setsockopt(sock, IPPROTO_IPV6, IPV6_MULTICAST_HOPS, &val, sizeof(val));
-int setup_dhcpv6_interface(struct interface *iface, bool enable)
-{
- // Configure multicast settings
- struct ipv6_mreq relay = {ALL_DHCPV6_RELAYS, iface->ifindex};
- struct ipv6_mreq server = {ALL_DHCPV6_SERVERS, iface->ifindex};
+ val = 0;
+ setsockopt(sock, IPPROTO_IPV6, IPV6_MULTICAST_LOOP, &val, sizeof(val));
- setsockopt(dhcpv6_event.uloop.fd, IPPROTO_IPV6,
- IPV6_DROP_MEMBERSHIP, &relay, sizeof(relay));
- setsockopt(dhcpv6_event.uloop.fd, IPPROTO_IPV6,
- IPV6_DROP_MEMBERSHIP, &server, sizeof(server));
+ struct sockaddr_in6 bind_addr = {AF_INET6, htons(DHCPV6_SERVER_PORT),
+ 0, IN6ADDR_ANY_INIT, 0};
- if (enable && iface->dhcpv6 && !iface->master) {
- setsockopt(dhcpv6_event.uloop.fd, IPPROTO_IPV6,
+ if (bind(sock, (struct sockaddr*)&bind_addr, sizeof(bind_addr))) {
+ syslog(LOG_ERR, "Failed to open DHCPv6 server socket: %s",
+ strerror(errno));
+ return -1;
+ }
+
+ struct ipv6_mreq relay = {ALL_DHCPV6_RELAYS, iface->ifindex};
+ struct ipv6_mreq server = {ALL_DHCPV6_SERVERS, iface->ifindex};
+ setsockopt(iface->dhcpv6_event.uloop.fd, IPPROTO_IPV6,
IPV6_ADD_MEMBERSHIP, &relay, sizeof(relay));
if (iface->dhcpv6 == RELAYD_SERVER)
- setsockopt(dhcpv6_event.uloop.fd, IPPROTO_IPV6,
+ setsockopt(iface->dhcpv6_event.uloop.fd, IPPROTO_IPV6,
IPV6_ADD_MEMBERSHIP, &server, sizeof(server));
+
+ iface->dhcpv6_event.uloop.fd = sock;
+ iface->dhcpv6_event.handle_dgram = handle_dhcpv6;
+ odhcpd_register(&iface->dhcpv6_event);
}
return setup_dhcpv6_ia_interface(iface, enable);
@@ -291,7 +291,7 @@ static void handle_client_request(void *addr, void *data, size_t len,
iov[3].iov_len + iov[4].iov_len + iov[5].iov_len +
iov[6].iov_len - (4 + opts_end - opts));
- odhcpd_send(dhcpv6_event.uloop.fd, addr, iov, ARRAY_SIZE(iov), iface);
+ odhcpd_send(iface->dhcpv6_event.uloop.fd, addr, iov, ARRAY_SIZE(iov), iface);
}
@@ -395,7 +395,7 @@ static void relay_server_response(uint8_t *data, size_t len)
}
struct iovec iov = {payload_data, payload_len};
- odhcpd_send(dhcpv6_event.uloop.fd, &target, &iov, 1, iface);
+ odhcpd_send(iface->dhcpv6_event.uloop.fd, &target, &iov, 1, iface);
}
@@ -452,5 +452,5 @@ static void relay_client_request(struct sockaddr_in6 *source,
struct sockaddr_in6 dhcpv6_servers = {AF_INET6,
htons(DHCPV6_SERVER_PORT), 0, ALL_DHCPV6_SERVERS, 0};
struct iovec iov[2] = {{&hdr, sizeof(hdr)}, {(void*)data, len}};
- odhcpd_send(dhcpv6_event.uloop.fd, &dhcpv6_servers, iov, 2, master);
+ odhcpd_send(iface->dhcpv6_event.uloop.fd, &dhcpv6_servers, iov, 2, master);
}
diff --git a/src/dhcpv6.h b/src/dhcpv6.h
index 693bc57..7b3b484 100644
--- a/src/dhcpv6.h
+++ b/src/dhcpv6.h
@@ -166,6 +166,6 @@ struct dhcpv6_assignment {
int dhcpv6_init_ia(struct interface *iface, int socket);
size_t dhcpv6_handle_ia(uint8_t *buf, size_t buflen, struct interface *iface,
const struct sockaddr_in6 *addr, const void *data, const uint8_t *end);
-int dhcpv6_ia_init(int dhcpv6_socket);
+int dhcpv6_ia_init(void);
int setup_dhcpv6_ia_interface(struct interface *iface, bool enable);
void dhcpv6_write_statefile(void);
diff --git a/src/odhcpd.h b/src/odhcpd.h
index 77376cb..789f696 100644
--- a/src/odhcpd.h
+++ b/src/odhcpd.h
@@ -115,6 +115,7 @@ struct interface {
bool ia_reconf;
// DHCPv4
+ struct odhcpd_event dhcpv6_event;
struct odhcpd_event dhcpv4_event;
struct list_head dhcpv4_assignments;