2013-12-09 22:43:08 +01:00
|
|
|
|
/***
|
|
|
|
|
This file is part of systemd.
|
|
|
|
|
|
|
|
|
|
Copyright (C) 2013 Intel Corporation. All rights reserved.
|
|
|
|
|
|
|
|
|
|
systemd is free software; you can redistribute it and/or modify it
|
|
|
|
|
under the terms of the GNU Lesser General Public License as published by
|
|
|
|
|
the Free Software Foundation; either version 2.1 of the License, or
|
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
|
|
systemd is distributed in the hope that it will be useful, but
|
|
|
|
|
WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
|
Lesser General Public License for more details.
|
|
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Lesser General Public License
|
|
|
|
|
along with systemd; If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
***/
|
|
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <stdio.h>
|
2013-12-09 22:43:19 +01:00
|
|
|
|
#include <net/ethernet.h>
|
2014-04-06 14:05:32 +02:00
|
|
|
|
#include <net/if_arp.h>
|
2014-04-20 19:10:02 +02:00
|
|
|
|
#include <netinet/ether.h>
|
2013-12-20 16:16:18 +01:00
|
|
|
|
#include <sys/param.h>
|
2014-02-23 17:30:13 +01:00
|
|
|
|
#include <sys/ioctl.h>
|
2013-12-09 22:43:08 +01:00
|
|
|
|
|
|
|
|
|
#include "util.h"
|
|
|
|
|
#include "list.h"
|
2014-04-09 12:12:07 +02:00
|
|
|
|
#include "refcnt.h"
|
2014-05-06 22:59:22 +02:00
|
|
|
|
#include "async.h"
|
2013-12-09 22:43:08 +01:00
|
|
|
|
|
|
|
|
|
#include "dhcp-protocol.h"
|
2013-12-09 22:43:19 +01:00
|
|
|
|
#include "dhcp-internal.h"
|
2014-02-27 01:24:05 +01:00
|
|
|
|
#include "dhcp-lease-internal.h"
|
2013-12-09 22:43:08 +01:00
|
|
|
|
#include "sd-dhcp-client.h"
|
|
|
|
|
|
|
|
|
|
struct sd_dhcp_client {
|
2014-04-09 12:12:07 +02:00
|
|
|
|
RefCount n_ref;
|
|
|
|
|
|
2013-12-09 22:43:08 +01:00
|
|
|
|
DHCPState state;
|
2013-12-09 22:43:25 +01:00
|
|
|
|
sd_event *event;
|
2014-01-18 19:32:45 +01:00
|
|
|
|
int event_priority;
|
2013-12-09 22:43:25 +01:00
|
|
|
|
sd_event_source *timeout_resend;
|
2013-12-09 22:43:08 +01:00
|
|
|
|
int index;
|
2013-12-09 22:43:26 +01:00
|
|
|
|
int fd;
|
|
|
|
|
union sockaddr_union link;
|
|
|
|
|
sd_event_source *receive_message;
|
2014-07-15 18:55:31 +02:00
|
|
|
|
bool request_broadcast;
|
2013-12-09 22:43:08 +01:00
|
|
|
|
uint8_t *req_opts;
|
2013-12-31 17:57:38 +01:00
|
|
|
|
size_t req_opts_allocated;
|
2013-12-09 22:43:08 +01:00
|
|
|
|
size_t req_opts_size;
|
2013-12-20 16:16:10 +01:00
|
|
|
|
be32_t last_addr;
|
2014-03-19 12:53:02 +01:00
|
|
|
|
struct {
|
|
|
|
|
uint8_t type;
|
|
|
|
|
struct ether_addr mac_addr;
|
|
|
|
|
} _packed_ client_id;
|
2014-07-01 20:58:49 +02:00
|
|
|
|
char *hostname;
|
2014-07-14 10:04:18 +02:00
|
|
|
|
char *vendor_class_identifier;
|
2013-12-09 22:43:19 +01:00
|
|
|
|
uint32_t xid;
|
2013-12-09 22:43:25 +01:00
|
|
|
|
usec_t start_time;
|
2014-01-31 10:31:25 +01:00
|
|
|
|
uint16_t secs;
|
2013-12-09 22:43:27 +01:00
|
|
|
|
unsigned int attempt;
|
2013-12-09 22:43:30 +01:00
|
|
|
|
usec_t request_sent;
|
|
|
|
|
sd_event_source *timeout_t1;
|
|
|
|
|
sd_event_source *timeout_t2;
|
|
|
|
|
sd_event_source *timeout_expire;
|
2013-12-09 22:43:31 +01:00
|
|
|
|
sd_dhcp_client_cb_t cb;
|
|
|
|
|
void *userdata;
|
2014-02-04 23:13:52 +01:00
|
|
|
|
sd_dhcp_lease *lease;
|
2013-12-09 22:43:08 +01:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static const uint8_t default_req_opts[] = {
|
|
|
|
|
DHCP_OPTION_SUBNET_MASK,
|
|
|
|
|
DHCP_OPTION_ROUTER,
|
|
|
|
|
DHCP_OPTION_HOST_NAME,
|
|
|
|
|
DHCP_OPTION_DOMAIN_NAME,
|
|
|
|
|
DHCP_OPTION_DOMAIN_NAME_SERVER,
|
|
|
|
|
DHCP_OPTION_NTP_SERVER,
|
|
|
|
|
};
|
|
|
|
|
|
2014-02-13 20:56:16 +01:00
|
|
|
|
static int client_receive_message_raw(sd_event_source *s, int fd,
|
|
|
|
|
uint32_t revents, void *userdata);
|
|
|
|
|
static int client_receive_message_udp(sd_event_source *s, int fd,
|
|
|
|
|
uint32_t revents, void *userdata);
|
2014-05-22 15:18:28 +02:00
|
|
|
|
static void client_stop(sd_dhcp_client *client, int error);
|
2013-12-20 16:16:18 +01:00
|
|
|
|
|
2013-12-09 22:43:31 +01:00
|
|
|
|
int sd_dhcp_client_set_callback(sd_dhcp_client *client, sd_dhcp_client_cb_t cb,
|
2014-01-16 19:55:25 +01:00
|
|
|
|
void *userdata) {
|
2013-12-09 22:43:31 +01:00
|
|
|
|
assert_return(client, -EINVAL);
|
|
|
|
|
|
|
|
|
|
client->cb = cb;
|
|
|
|
|
client->userdata = userdata;
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-07-15 18:55:31 +02:00
|
|
|
|
int sd_dhcp_client_set_request_broadcast(sd_dhcp_client *client, int broadcast) {
|
|
|
|
|
assert_return(client, -EINVAL);
|
|
|
|
|
|
|
|
|
|
client->request_broadcast = !!broadcast;
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-01-16 19:55:25 +01:00
|
|
|
|
int sd_dhcp_client_set_request_option(sd_dhcp_client *client, uint8_t option) {
|
2013-12-09 22:43:08 +01:00
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
|
|
assert_return(client, -EINVAL);
|
2014-04-09 12:12:08 +02:00
|
|
|
|
assert_return (IN_SET(client->state, DHCP_STATE_INIT,
|
|
|
|
|
DHCP_STATE_STOPPED), -EBUSY);
|
2013-12-09 22:43:08 +01:00
|
|
|
|
|
|
|
|
|
switch(option) {
|
|
|
|
|
case DHCP_OPTION_PAD:
|
|
|
|
|
case DHCP_OPTION_OVERLOAD:
|
|
|
|
|
case DHCP_OPTION_MESSAGE_TYPE:
|
|
|
|
|
case DHCP_OPTION_PARAMETER_REQUEST_LIST:
|
|
|
|
|
case DHCP_OPTION_END:
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < client->req_opts_size; i++)
|
|
|
|
|
if (client->req_opts[i] == option)
|
|
|
|
|
return -EEXIST;
|
|
|
|
|
|
2013-12-31 17:57:38 +01:00
|
|
|
|
if (!GREEDY_REALLOC(client->req_opts, client->req_opts_allocated,
|
2013-12-09 22:43:08 +01:00
|
|
|
|
client->req_opts_size + 1))
|
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
2013-12-31 17:57:38 +01:00
|
|
|
|
client->req_opts[client->req_opts_size++] = option;
|
2013-12-09 22:43:08 +01:00
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int sd_dhcp_client_set_request_address(sd_dhcp_client *client,
|
2014-01-16 19:55:25 +01:00
|
|
|
|
const struct in_addr *last_addr) {
|
2013-12-09 22:43:08 +01:00
|
|
|
|
assert_return(client, -EINVAL);
|
2014-04-09 12:12:08 +02:00
|
|
|
|
assert_return (IN_SET(client->state, DHCP_STATE_INIT,
|
|
|
|
|
DHCP_STATE_STOPPED), -EBUSY);
|
2013-12-09 22:43:08 +01:00
|
|
|
|
|
|
|
|
|
if (last_addr)
|
|
|
|
|
client->last_addr = last_addr->s_addr;
|
|
|
|
|
else
|
|
|
|
|
client->last_addr = INADDR_ANY;
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-01-16 19:55:25 +01:00
|
|
|
|
int sd_dhcp_client_set_index(sd_dhcp_client *client, int interface_index) {
|
2013-12-09 22:43:08 +01:00
|
|
|
|
assert_return(client, -EINVAL);
|
2014-04-09 12:12:08 +02:00
|
|
|
|
assert_return (IN_SET(client->state, DHCP_STATE_INIT,
|
|
|
|
|
DHCP_STATE_STOPPED), -EBUSY);
|
2014-04-27 21:58:26 +02:00
|
|
|
|
assert_return(interface_index > 0, -EINVAL);
|
2013-12-09 22:43:08 +01:00
|
|
|
|
|
|
|
|
|
client->index = interface_index;
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-09 22:43:19 +01:00
|
|
|
|
int sd_dhcp_client_set_mac(sd_dhcp_client *client,
|
2014-01-16 19:55:25 +01:00
|
|
|
|
const struct ether_addr *addr) {
|
2014-05-22 15:18:28 +02:00
|
|
|
|
DHCP_CLIENT_DONT_DESTROY(client);
|
2014-03-21 18:36:32 +01:00
|
|
|
|
bool need_restart = false;
|
|
|
|
|
|
2013-12-09 22:43:19 +01:00
|
|
|
|
assert_return(client, -EINVAL);
|
2014-03-21 18:36:32 +01:00
|
|
|
|
assert_return(addr, -EINVAL);
|
2013-12-09 22:43:19 +01:00
|
|
|
|
|
2014-03-21 18:36:32 +01:00
|
|
|
|
if (memcmp(&client->client_id.mac_addr, addr, ETH_ALEN) == 0)
|
|
|
|
|
return 0;
|
|
|
|
|
|
2014-04-09 12:12:08 +02:00
|
|
|
|
if (!IN_SET(client->state, DHCP_STATE_INIT, DHCP_STATE_STOPPED)) {
|
2014-03-21 18:36:32 +01:00
|
|
|
|
log_dhcp_client(client, "Changing MAC address on running DHCP "
|
|
|
|
|
"client, restarting");
|
|
|
|
|
need_restart = true;
|
2014-05-22 15:18:28 +02:00
|
|
|
|
client_stop(client, DHCP_EVENT_STOP);
|
2014-03-21 18:36:32 +01:00
|
|
|
|
}
|
2014-03-05 11:07:15 +01:00
|
|
|
|
|
2014-03-19 12:53:02 +01:00
|
|
|
|
memcpy(&client->client_id.mac_addr, addr, ETH_ALEN);
|
|
|
|
|
client->client_id.type = 0x01;
|
2013-12-09 22:43:19 +01:00
|
|
|
|
|
2014-04-09 12:12:08 +02:00
|
|
|
|
if (need_restart && client->state != DHCP_STATE_STOPPED)
|
2014-03-21 18:36:32 +01:00
|
|
|
|
sd_dhcp_client_start(client);
|
|
|
|
|
|
2013-12-09 22:43:19 +01:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-07-01 20:58:49 +02:00
|
|
|
|
int sd_dhcp_client_set_hostname(sd_dhcp_client *client,
|
|
|
|
|
const char *hostname) {
|
|
|
|
|
char *new_hostname = NULL;
|
|
|
|
|
|
|
|
|
|
assert_return(client, -EINVAL);
|
|
|
|
|
|
|
|
|
|
if (streq_ptr(client->hostname, hostname))
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
if (hostname) {
|
|
|
|
|
new_hostname = strdup(hostname);
|
|
|
|
|
if (!new_hostname)
|
|
|
|
|
return -ENOMEM;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
free(client->hostname);
|
|
|
|
|
client->hostname = new_hostname;
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-07-14 10:04:18 +02:00
|
|
|
|
int sd_dhcp_client_set_vendor_class_identifier(sd_dhcp_client *client,
|
|
|
|
|
const char *vci) {
|
|
|
|
|
char *new_vci = NULL;
|
|
|
|
|
|
|
|
|
|
assert_return(client, -EINVAL);
|
|
|
|
|
|
|
|
|
|
new_vci = strdup(vci);
|
|
|
|
|
if (!new_vci)
|
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
|
|
free(client->vendor_class_identifier);
|
|
|
|
|
|
|
|
|
|
client->vendor_class_identifier = new_vci;
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-02-04 23:13:52 +01:00
|
|
|
|
int sd_dhcp_client_get_lease(sd_dhcp_client *client, sd_dhcp_lease **ret) {
|
2013-12-09 22:43:31 +01:00
|
|
|
|
assert_return(client, -EINVAL);
|
2014-02-04 23:13:52 +01:00
|
|
|
|
assert_return(ret, -EINVAL);
|
2013-12-09 22:43:31 +01:00
|
|
|
|
|
2014-02-04 23:13:52 +01:00
|
|
|
|
if (client->state != DHCP_STATE_BOUND &&
|
|
|
|
|
client->state != DHCP_STATE_RENEWING &&
|
|
|
|
|
client->state != DHCP_STATE_REBINDING)
|
|
|
|
|
return -EADDRNOTAVAIL;
|
2013-12-09 22:43:31 +01:00
|
|
|
|
|
2014-02-04 23:13:52 +01:00
|
|
|
|
*ret = sd_dhcp_lease_ref(client->lease);
|
2014-01-13 23:07:59 +01:00
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-22 15:18:28 +02:00
|
|
|
|
static void client_notify(sd_dhcp_client *client, int event) {
|
|
|
|
|
if (client->cb)
|
2013-12-09 22:43:31 +01:00
|
|
|
|
client->cb(client, event, client->userdata);
|
2013-12-09 22:43:29 +01:00
|
|
|
|
}
|
|
|
|
|
|
2014-03-12 10:46:40 +01:00
|
|
|
|
static int client_initialize(sd_dhcp_client *client) {
|
2013-12-09 22:43:23 +01:00
|
|
|
|
assert_return(client, -EINVAL);
|
|
|
|
|
|
2013-12-09 22:43:26 +01:00
|
|
|
|
client->receive_message =
|
|
|
|
|
sd_event_source_unref(client->receive_message);
|
|
|
|
|
|
2014-05-06 22:59:22 +02:00
|
|
|
|
client->fd = asynchronous_close(client->fd);
|
2013-12-09 22:43:26 +01:00
|
|
|
|
|
2013-12-09 22:43:25 +01:00
|
|
|
|
client->timeout_resend = sd_event_source_unref(client->timeout_resend);
|
|
|
|
|
|
2013-12-09 22:43:30 +01:00
|
|
|
|
client->timeout_t1 = sd_event_source_unref(client->timeout_t1);
|
|
|
|
|
client->timeout_t2 = sd_event_source_unref(client->timeout_t2);
|
|
|
|
|
client->timeout_expire = sd_event_source_unref(client->timeout_expire);
|
|
|
|
|
|
2013-12-09 22:43:27 +01:00
|
|
|
|
client->attempt = 1;
|
|
|
|
|
|
2014-01-31 10:31:24 +01:00
|
|
|
|
client->state = DHCP_STATE_INIT;
|
2014-03-12 10:46:40 +01:00
|
|
|
|
client->xid = 0;
|
2013-12-09 22:43:23 +01:00
|
|
|
|
|
2014-02-04 23:13:52 +01:00
|
|
|
|
if (client->lease)
|
|
|
|
|
client->lease = sd_dhcp_lease_unref(client->lease);
|
2013-12-09 22:43:26 +01:00
|
|
|
|
|
2014-03-12 10:46:40 +01:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-22 15:18:28 +02:00
|
|
|
|
static void client_stop(sd_dhcp_client *client, int error) {
|
|
|
|
|
assert(client);
|
2014-03-12 10:46:40 +01:00
|
|
|
|
|
2014-04-27 22:01:42 +02:00
|
|
|
|
if (error < 0)
|
|
|
|
|
log_dhcp_client(client, "STOPPED: %s", strerror(-error));
|
|
|
|
|
else {
|
|
|
|
|
switch(error) {
|
|
|
|
|
case DHCP_EVENT_STOP:
|
2014-05-09 00:26:22 +02:00
|
|
|
|
log_dhcp_client(client, "STOPPED");
|
2014-04-27 22:01:42 +02:00
|
|
|
|
break;
|
|
|
|
|
case DHCP_EVENT_NO_LEASE:
|
|
|
|
|
log_dhcp_client(client, "STOPPED: No lease");
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
log_dhcp_client(client, "STOPPED: Unknown reason");
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2014-03-12 10:46:40 +01:00
|
|
|
|
|
2014-05-22 15:18:28 +02:00
|
|
|
|
client_notify(client, error);
|
2014-02-22 19:53:45 +01:00
|
|
|
|
|
2014-05-22 15:18:28 +02:00
|
|
|
|
client_initialize(client);
|
2013-12-09 22:43:23 +01:00
|
|
|
|
}
|
|
|
|
|
|
2014-05-21 16:46:14 +02:00
|
|
|
|
static int client_message_init(sd_dhcp_client *client, DHCPPacket **ret,
|
|
|
|
|
uint8_t type, size_t *_optlen, size_t *_optoffset) {
|
|
|
|
|
_cleanup_free_ DHCPPacket *packet;
|
|
|
|
|
size_t optlen, optoffset, size;
|
2014-04-11 19:54:04 +02:00
|
|
|
|
be16_t max_size;
|
2014-02-11 13:11:18 +01:00
|
|
|
|
int r;
|
2013-12-09 22:43:19 +01:00
|
|
|
|
|
2014-03-19 14:45:35 +01:00
|
|
|
|
assert(client);
|
|
|
|
|
assert(client->secs);
|
2014-05-21 16:46:14 +02:00
|
|
|
|
assert(ret);
|
|
|
|
|
assert(_optlen);
|
|
|
|
|
assert(_optoffset);
|
2014-04-11 19:54:04 +02:00
|
|
|
|
assert(type == DHCP_DISCOVER || type == DHCP_REQUEST);
|
2014-02-23 22:07:07 +01:00
|
|
|
|
|
2014-05-21 16:46:14 +02:00
|
|
|
|
optlen = DHCP_MIN_OPTIONS_SIZE;
|
|
|
|
|
size = sizeof(DHCPPacket) + optlen;
|
|
|
|
|
|
|
|
|
|
packet = malloc0(size);
|
|
|
|
|
if (!packet)
|
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
|
|
r = dhcp_message_init(&packet->dhcp, BOOTREQUEST, client->xid, type,
|
|
|
|
|
optlen, &optoffset);
|
2014-02-11 13:11:18 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
2013-12-09 22:43:19 +01:00
|
|
|
|
|
2014-02-23 22:07:07 +01:00
|
|
|
|
/* Although 'secs' field is a SHOULD in RFC 2131, certain DHCP servers
|
|
|
|
|
refuse to issue an DHCP lease if 'secs' is set to zero */
|
2014-05-21 16:46:14 +02:00
|
|
|
|
packet->dhcp.secs = htobe16(client->secs);
|
2014-02-23 22:07:07 +01:00
|
|
|
|
|
2014-05-28 20:43:37 +02:00
|
|
|
|
/* RFC2132 section 4.1
|
|
|
|
|
A client that cannot receive unicast IP datagrams until its protocol
|
|
|
|
|
software has been configured with an IP address SHOULD set the
|
|
|
|
|
BROADCAST bit in the 'flags' field to 1 in any DHCPDISCOVER or
|
|
|
|
|
DHCPREQUEST messages that client sends. The BROADCAST bit will
|
|
|
|
|
provide a hint to the DHCP server and BOOTP relay agent to broadcast
|
2014-07-15 18:55:31 +02:00
|
|
|
|
any messages to the client on the client's subnet.
|
|
|
|
|
|
|
|
|
|
Note: some interfaces needs this to be enabled, but some networks
|
|
|
|
|
needs this to be disabled as broadcasts are filteretd, so this
|
|
|
|
|
needs to be configurable */
|
|
|
|
|
if (client->request_broadcast)
|
|
|
|
|
packet->dhcp.flags = htobe16(0x8000);
|
2014-05-28 20:43:37 +02:00
|
|
|
|
|
2014-04-11 21:02:16 +02:00
|
|
|
|
/* RFC2132 section 4.1.1:
|
|
|
|
|
The client MUST include its hardware address in the ’chaddr’ field, if
|
|
|
|
|
necessary for delivery of DHCP reply messages.
|
|
|
|
|
*/
|
2014-05-21 16:46:14 +02:00
|
|
|
|
memcpy(&packet->dhcp.chaddr, &client->client_id.mac_addr, ETH_ALEN);
|
2013-12-09 22:43:22 +01:00
|
|
|
|
|
2014-02-22 19:53:45 +01:00
|
|
|
|
/* Some DHCP servers will refuse to issue an DHCP lease if the Client
|
2013-12-09 22:43:19 +01:00
|
|
|
|
Identifier option is not set */
|
2014-05-21 16:46:14 +02:00
|
|
|
|
r = dhcp_option_append(&packet->dhcp, optlen, &optoffset, 0,
|
2014-05-20 11:04:50 +02:00
|
|
|
|
DHCP_OPTION_CLIENT_IDENTIFIER,
|
2014-03-19 12:53:02 +01:00
|
|
|
|
sizeof(client->client_id), &client->client_id);
|
2014-02-11 13:11:18 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
2013-12-09 22:43:19 +01:00
|
|
|
|
|
2014-04-11 21:02:16 +02:00
|
|
|
|
|
|
|
|
|
/* RFC2131 section 3.5:
|
|
|
|
|
in its initial DHCPDISCOVER or DHCPREQUEST message, a
|
|
|
|
|
client may provide the server with a list of specific
|
|
|
|
|
parameters the client is interested in. If the client
|
|
|
|
|
includes a list of parameters in a DHCPDISCOVER message,
|
|
|
|
|
it MUST include that list in any subsequent DHCPREQUEST
|
|
|
|
|
messages.
|
|
|
|
|
*/
|
2014-05-21 16:46:14 +02:00
|
|
|
|
r = dhcp_option_append(&packet->dhcp, optlen, &optoffset, 0,
|
2014-04-11 19:54:04 +02:00
|
|
|
|
DHCP_OPTION_PARAMETER_REQUEST_LIST,
|
2014-05-20 11:04:50 +02:00
|
|
|
|
client->req_opts_size, client->req_opts);
|
2014-04-11 19:54:04 +02:00
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
|
|
|
|
|
2014-04-11 21:02:16 +02:00
|
|
|
|
/* RFC2131 section 3.5:
|
|
|
|
|
The client SHOULD include the ’maximum DHCP message size’ option to
|
|
|
|
|
let the server know how large the server may make its DHCP messages.
|
|
|
|
|
|
|
|
|
|
Note (from ConnMan): Some DHCP servers will send bigger DHCP packets
|
|
|
|
|
than the defined default size unless the Maximum Messge Size option
|
|
|
|
|
is explicitely set
|
|
|
|
|
*/
|
2014-05-21 16:46:14 +02:00
|
|
|
|
max_size = htobe16(size);
|
|
|
|
|
r = dhcp_option_append(&packet->dhcp, optlen, &optoffset, 0,
|
2014-04-11 19:54:04 +02:00
|
|
|
|
DHCP_OPTION_MAXIMUM_MESSAGE_SIZE,
|
|
|
|
|
2, &max_size);
|
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
2013-12-09 22:43:19 +01:00
|
|
|
|
|
2014-05-21 16:46:14 +02:00
|
|
|
|
*_optlen = optlen;
|
|
|
|
|
*_optoffset = optoffset;
|
|
|
|
|
*ret = packet;
|
|
|
|
|
packet = NULL;
|
|
|
|
|
|
2013-12-09 22:43:19 +01:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-03-09 22:51:07 +01:00
|
|
|
|
static int dhcp_client_send_raw(sd_dhcp_client *client, DHCPPacket *packet,
|
|
|
|
|
size_t len) {
|
|
|
|
|
dhcp_packet_append_ip_headers(packet, INADDR_ANY, DHCP_PORT_CLIENT,
|
|
|
|
|
INADDR_BROADCAST, DHCP_PORT_SERVER, len);
|
|
|
|
|
|
|
|
|
|
return dhcp_network_send_raw_socket(client->fd, &client->link,
|
|
|
|
|
packet, len);
|
|
|
|
|
}
|
|
|
|
|
|
2014-03-19 14:45:35 +01:00
|
|
|
|
static int client_send_discover(sd_dhcp_client *client) {
|
2014-03-27 20:13:11 +01:00
|
|
|
|
_cleanup_free_ DHCPPacket *discover = NULL;
|
2014-05-21 16:46:14 +02:00
|
|
|
|
size_t optoffset, optlen;
|
2014-03-19 14:45:35 +01:00
|
|
|
|
usec_t time_now;
|
|
|
|
|
int r;
|
|
|
|
|
|
|
|
|
|
assert(client);
|
2014-04-11 21:02:16 +02:00
|
|
|
|
assert(client->state == DHCP_STATE_INIT ||
|
|
|
|
|
client->state == DHCP_STATE_SELECTING);
|
|
|
|
|
|
|
|
|
|
/* See RFC2131 section 4.4.1 */
|
2014-03-19 14:45:35 +01:00
|
|
|
|
|
2014-03-24 02:49:09 +01:00
|
|
|
|
r = sd_event_now(client->event, CLOCK_MONOTONIC, &time_now);
|
2014-03-19 14:45:35 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
|
|
|
|
assert(time_now >= client->start_time);
|
|
|
|
|
|
|
|
|
|
/* seconds between sending first and last DISCOVER
|
|
|
|
|
* must always be strictly positive to deal with broken servers */
|
|
|
|
|
client->secs = ((time_now - client->start_time) / USEC_PER_SEC) ? : 1;
|
2013-12-09 22:43:19 +01:00
|
|
|
|
|
2014-05-21 16:46:14 +02:00
|
|
|
|
r = client_message_init(client, &discover, DHCP_DISCOVER,
|
|
|
|
|
&optlen, &optoffset);
|
2014-03-19 14:45:35 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
2013-12-09 22:43:19 +01:00
|
|
|
|
|
2014-04-11 21:02:16 +02:00
|
|
|
|
/* the client may suggest values for the network address
|
|
|
|
|
and lease time in the DHCPDISCOVER message. The client may include
|
|
|
|
|
the ’requested IP address’ option to suggest that a particular IP
|
|
|
|
|
address be assigned, and may include the ’IP address lease time’
|
|
|
|
|
option to suggest the lease time it would like.
|
|
|
|
|
*/
|
2013-12-09 22:43:19 +01:00
|
|
|
|
if (client->last_addr != INADDR_ANY) {
|
2014-05-21 15:55:02 +02:00
|
|
|
|
r = dhcp_option_append(&discover->dhcp, optlen, &optoffset, 0,
|
2014-05-20 11:04:50 +02:00
|
|
|
|
DHCP_OPTION_REQUESTED_IP_ADDRESS,
|
|
|
|
|
4, &client->last_addr);
|
2014-03-19 14:45:35 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
2013-12-09 22:43:19 +01:00
|
|
|
|
}
|
|
|
|
|
|
2014-07-01 20:58:49 +02:00
|
|
|
|
/* it is unclear from RFC 2131 if client should send hostname in
|
|
|
|
|
DHCPDISCOVER but dhclient does and so we do as well
|
|
|
|
|
*/
|
|
|
|
|
if (client->hostname) {
|
|
|
|
|
r = dhcp_option_append(&discover->dhcp, optlen, &optoffset, 0,
|
|
|
|
|
DHCP_OPTION_HOST_NAME,
|
|
|
|
|
strlen(client->hostname), client->hostname);
|
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
|
|
|
|
}
|
|
|
|
|
|
2014-07-14 10:04:18 +02:00
|
|
|
|
if (client->vendor_class_identifier) {
|
|
|
|
|
r = dhcp_option_append(&discover->dhcp, optlen, &optoffset, 0,
|
|
|
|
|
DHCP_OPTION_VENDOR_CLASS_IDENTIFIER,
|
|
|
|
|
strlen(client->vendor_class_identifier),
|
|
|
|
|
client->vendor_class_identifier);
|
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-21 15:55:02 +02:00
|
|
|
|
r = dhcp_option_append(&discover->dhcp, optlen, &optoffset, 0,
|
2014-05-20 11:04:50 +02:00
|
|
|
|
DHCP_OPTION_END, 0, NULL);
|
2014-06-18 20:26:54 +02:00
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
2013-12-09 22:43:19 +01:00
|
|
|
|
|
2014-04-11 21:02:16 +02:00
|
|
|
|
/* We currently ignore:
|
|
|
|
|
The client SHOULD wait a random time between one and ten seconds to
|
|
|
|
|
desynchronize the use of DHCP at startup.
|
|
|
|
|
*/
|
2014-05-20 11:04:50 +02:00
|
|
|
|
r = dhcp_client_send_raw(client, discover, sizeof(DHCPPacket) + optoffset);
|
2014-03-19 14:45:35 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
2013-12-09 22:43:27 +01:00
|
|
|
|
|
2014-02-22 19:53:45 +01:00
|
|
|
|
log_dhcp_client(client, "DISCOVER");
|
|
|
|
|
|
2014-03-09 22:51:07 +01:00
|
|
|
|
return 0;
|
2013-12-09 22:43:27 +01:00
|
|
|
|
}
|
|
|
|
|
|
2014-03-19 14:45:35 +01:00
|
|
|
|
static int client_send_request(sd_dhcp_client *client) {
|
2014-06-13 18:48:20 +02:00
|
|
|
|
_cleanup_free_ DHCPPacket *request = NULL;
|
2014-05-21 16:46:14 +02:00
|
|
|
|
size_t optoffset, optlen;
|
2014-03-19 14:45:35 +01:00
|
|
|
|
int r;
|
2013-12-09 22:43:27 +01:00
|
|
|
|
|
2014-05-21 16:46:14 +02:00
|
|
|
|
r = client_message_init(client, &request, DHCP_REQUEST,
|
|
|
|
|
&optlen, &optoffset);
|
2014-03-19 14:45:35 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
2013-12-09 22:43:19 +01:00
|
|
|
|
|
2014-01-31 10:31:22 +01:00
|
|
|
|
switch (client->state) {
|
2014-04-11 21:02:16 +02:00
|
|
|
|
/* See RFC2131 section 4.3.2 (note that there is a typo in the RFC,
|
|
|
|
|
SELECTING should be REQUESTING)
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
case DHCP_STATE_REQUESTING:
|
|
|
|
|
/* Client inserts the address of the selected server in ’server
|
|
|
|
|
identifier’, ’ciaddr’ MUST be zero, ’requested IP address’ MUST be
|
|
|
|
|
filled in with the yiaddr value from the chosen DHCPOFFER.
|
|
|
|
|
*/
|
2014-01-31 10:31:22 +01:00
|
|
|
|
|
2014-05-21 15:55:02 +02:00
|
|
|
|
r = dhcp_option_append(&request->dhcp, optlen, &optoffset, 0,
|
2014-04-11 21:02:16 +02:00
|
|
|
|
DHCP_OPTION_SERVER_IDENTIFIER,
|
|
|
|
|
4, &client->lease->server_address);
|
2014-03-19 14:45:35 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
2014-01-31 10:31:22 +01:00
|
|
|
|
|
2014-05-21 15:55:02 +02:00
|
|
|
|
r = dhcp_option_append(&request->dhcp, optlen, &optoffset, 0,
|
2014-03-19 14:45:35 +01:00
|
|
|
|
DHCP_OPTION_REQUESTED_IP_ADDRESS,
|
|
|
|
|
4, &client->lease->address);
|
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
|
|
|
|
|
2014-04-11 21:02:16 +02:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case DHCP_STATE_INIT_REBOOT:
|
|
|
|
|
/* ’server identifier’ MUST NOT be filled in, ’requested IP address’
|
|
|
|
|
option MUST be filled in with client’s notion of its previously
|
|
|
|
|
assigned address. ’ciaddr’ MUST be zero.
|
|
|
|
|
*/
|
2014-05-21 15:55:02 +02:00
|
|
|
|
r = dhcp_option_append(&request->dhcp, optlen, &optoffset, 0,
|
2014-04-11 21:02:16 +02:00
|
|
|
|
DHCP_OPTION_REQUESTED_IP_ADDRESS,
|
|
|
|
|
4, &client->last_addr);
|
2014-03-19 14:45:35 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
2014-01-31 10:31:22 +01:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case DHCP_STATE_RENEWING:
|
2014-04-11 21:02:16 +02:00
|
|
|
|
/* ’server identifier’ MUST NOT be filled in, ’requested IP address’
|
|
|
|
|
option MUST NOT be filled in, ’ciaddr’ MUST be filled in with
|
|
|
|
|
client’s IP address.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/* fall through */
|
2014-01-31 10:31:22 +01:00
|
|
|
|
case DHCP_STATE_REBINDING:
|
2014-04-11 21:02:16 +02:00
|
|
|
|
/* ’server identifier’ MUST NOT be filled in, ’requested IP address’
|
|
|
|
|
option MUST NOT be filled in, ’ciaddr’ MUST be filled in with
|
|
|
|
|
client’s IP address.
|
|
|
|
|
|
|
|
|
|
This message MUST be broadcast to the 0xffffffff IP broadcast address.
|
|
|
|
|
*/
|
|
|
|
|
request->dhcp.ciaddr = client->lease->address;
|
2014-01-31 10:31:22 +01:00
|
|
|
|
|
|
|
|
|
break;
|
2014-04-09 12:12:08 +02:00
|
|
|
|
|
2014-04-11 21:02:16 +02:00
|
|
|
|
case DHCP_STATE_INIT:
|
|
|
|
|
case DHCP_STATE_SELECTING:
|
|
|
|
|
case DHCP_STATE_REBOOTING:
|
|
|
|
|
case DHCP_STATE_BOUND:
|
2014-04-09 12:12:08 +02:00
|
|
|
|
case DHCP_STATE_STOPPED:
|
|
|
|
|
return -EINVAL;
|
2013-12-09 22:43:27 +01:00
|
|
|
|
}
|
2013-12-09 22:43:19 +01:00
|
|
|
|
|
2014-07-01 20:58:49 +02:00
|
|
|
|
if (client->hostname) {
|
|
|
|
|
r = dhcp_option_append(&request->dhcp, optlen, &optoffset, 0,
|
|
|
|
|
DHCP_OPTION_HOST_NAME,
|
|
|
|
|
strlen(client->hostname), client->hostname);
|
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-21 15:55:02 +02:00
|
|
|
|
r = dhcp_option_append(&request->dhcp, optlen, &optoffset, 0,
|
2014-05-20 11:04:50 +02:00
|
|
|
|
DHCP_OPTION_END, 0, NULL);
|
2014-03-19 14:45:35 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
2013-12-09 22:43:19 +01:00
|
|
|
|
|
2013-12-20 16:16:18 +01:00
|
|
|
|
if (client->state == DHCP_STATE_RENEWING) {
|
2014-03-19 14:45:35 +01:00
|
|
|
|
r = dhcp_network_send_udp_socket(client->fd,
|
|
|
|
|
client->lease->server_address,
|
|
|
|
|
DHCP_PORT_SERVER,
|
|
|
|
|
&request->dhcp,
|
2014-05-20 11:04:50 +02:00
|
|
|
|
sizeof(DHCPMessage) + optoffset);
|
2013-12-20 16:16:18 +01:00
|
|
|
|
} else {
|
2014-05-20 11:04:50 +02:00
|
|
|
|
r = dhcp_client_send_raw(client, request, sizeof(DHCPPacket) + optoffset);
|
2013-12-20 16:16:18 +01:00
|
|
|
|
}
|
2014-03-19 14:45:35 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
2013-12-09 22:43:19 +01:00
|
|
|
|
|
2014-04-12 01:01:13 +02:00
|
|
|
|
switch (client->state) {
|
|
|
|
|
case DHCP_STATE_REQUESTING:
|
|
|
|
|
log_dhcp_client(client, "REQUEST (requesting)");
|
|
|
|
|
break;
|
|
|
|
|
case DHCP_STATE_INIT_REBOOT:
|
|
|
|
|
log_dhcp_client(client, "REQUEST (init-reboot)");
|
|
|
|
|
break;
|
|
|
|
|
case DHCP_STATE_RENEWING:
|
|
|
|
|
log_dhcp_client(client, "REQUEST (renewing)");
|
|
|
|
|
break;
|
|
|
|
|
case DHCP_STATE_REBINDING:
|
|
|
|
|
log_dhcp_client(client, "REQUEST (rebinding)");
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
log_dhcp_client(client, "REQUEST (invalid)");
|
|
|
|
|
break;
|
|
|
|
|
}
|
2014-02-22 19:53:45 +01:00
|
|
|
|
|
2014-03-09 22:51:07 +01:00
|
|
|
|
return 0;
|
2013-12-09 22:43:19 +01:00
|
|
|
|
}
|
|
|
|
|
|
2014-04-12 00:34:05 +02:00
|
|
|
|
static int client_start(sd_dhcp_client *client);
|
|
|
|
|
|
2013-12-09 22:43:25 +01:00
|
|
|
|
static int client_timeout_resend(sd_event_source *s, uint64_t usec,
|
2014-01-16 19:55:25 +01:00
|
|
|
|
void *userdata) {
|
2013-12-09 22:43:25 +01:00
|
|
|
|
sd_dhcp_client *client = userdata;
|
2014-05-22 15:18:28 +02:00
|
|
|
|
DHCP_CLIENT_DONT_DESTROY(client);
|
2013-12-20 16:16:18 +01:00
|
|
|
|
usec_t next_timeout = 0;
|
2014-03-18 14:13:01 +01:00
|
|
|
|
uint64_t time_now;
|
2013-12-20 16:16:18 +01:00
|
|
|
|
uint32_t time_left;
|
2014-03-18 14:13:01 +01:00
|
|
|
|
int r;
|
2014-01-18 19:32:45 +01:00
|
|
|
|
|
|
|
|
|
assert(s);
|
|
|
|
|
assert(client);
|
|
|
|
|
assert(client->event);
|
2013-12-09 22:43:25 +01:00
|
|
|
|
|
2014-03-24 02:49:09 +01:00
|
|
|
|
r = sd_event_now(client->event, CLOCK_MONOTONIC, &time_now);
|
2014-03-18 14:13:01 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
goto error;
|
|
|
|
|
|
2013-12-20 16:16:18 +01:00
|
|
|
|
switch (client->state) {
|
|
|
|
|
case DHCP_STATE_RENEWING:
|
|
|
|
|
|
2014-02-13 20:56:16 +01:00
|
|
|
|
time_left = (client->lease->t2 - client->lease->t1) / 2;
|
2013-12-20 16:16:18 +01:00
|
|
|
|
if (time_left < 60)
|
|
|
|
|
time_left = 60;
|
2013-12-09 22:43:25 +01:00
|
|
|
|
|
2014-03-18 14:13:01 +01:00
|
|
|
|
next_timeout = time_now + time_left * USEC_PER_SEC;
|
2013-12-09 22:43:25 +01:00
|
|
|
|
|
2013-12-20 16:16:18 +01:00
|
|
|
|
break;
|
|
|
|
|
|
2013-12-20 16:16:20 +01:00
|
|
|
|
case DHCP_STATE_REBINDING:
|
|
|
|
|
|
2014-02-13 20:56:16 +01:00
|
|
|
|
time_left = (client->lease->lifetime - client->lease->t2) / 2;
|
2013-12-20 16:16:20 +01:00
|
|
|
|
if (time_left < 60)
|
|
|
|
|
time_left = 60;
|
|
|
|
|
|
2014-03-18 14:13:01 +01:00
|
|
|
|
next_timeout = time_now + time_left * USEC_PER_SEC;
|
2013-12-20 16:16:20 +01:00
|
|
|
|
break;
|
|
|
|
|
|
2014-01-31 10:31:22 +01:00
|
|
|
|
case DHCP_STATE_REBOOTING:
|
|
|
|
|
/* start over as we did not receive a timely ack or nak */
|
2014-04-12 00:34:05 +02:00
|
|
|
|
r = client_initialize(client);
|
2014-04-11 18:02:54 +02:00
|
|
|
|
if (r < 0)
|
|
|
|
|
goto error;
|
2014-01-31 10:31:22 +01:00
|
|
|
|
|
2014-04-12 01:01:13 +02:00
|
|
|
|
r = client_start(client);
|
|
|
|
|
if (r < 0)
|
|
|
|
|
goto error;
|
|
|
|
|
else {
|
|
|
|
|
log_dhcp_client(client, "REBOOTED");
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2014-04-12 00:34:05 +02:00
|
|
|
|
|
2013-12-20 16:16:18 +01:00
|
|
|
|
case DHCP_STATE_INIT:
|
|
|
|
|
case DHCP_STATE_INIT_REBOOT:
|
|
|
|
|
case DHCP_STATE_SELECTING:
|
|
|
|
|
case DHCP_STATE_REQUESTING:
|
|
|
|
|
case DHCP_STATE_BOUND:
|
|
|
|
|
|
|
|
|
|
if (client->attempt < 64)
|
|
|
|
|
client->attempt *= 2;
|
|
|
|
|
|
2014-03-18 14:13:01 +01:00
|
|
|
|
next_timeout = time_now + (client->attempt - 1) * USEC_PER_SEC;
|
2013-12-20 16:16:18 +01:00
|
|
|
|
|
|
|
|
|
break;
|
2014-04-09 12:12:08 +02:00
|
|
|
|
|
|
|
|
|
case DHCP_STATE_STOPPED:
|
|
|
|
|
r = -EINVAL;
|
|
|
|
|
goto error;
|
2013-12-20 16:16:18 +01:00
|
|
|
|
}
|
|
|
|
|
|
2013-12-22 19:59:12 +01:00
|
|
|
|
next_timeout += (random_u32() & 0x1fffff);
|
2013-12-09 22:43:25 +01:00
|
|
|
|
|
2014-02-20 21:04:03 +01:00
|
|
|
|
client->timeout_resend = sd_event_source_unref(client->timeout_resend);
|
|
|
|
|
|
2014-03-24 02:49:09 +01:00
|
|
|
|
r = sd_event_add_time(client->event,
|
|
|
|
|
&client->timeout_resend,
|
|
|
|
|
CLOCK_MONOTONIC,
|
|
|
|
|
next_timeout, 10 * USEC_PER_MSEC,
|
|
|
|
|
client_timeout_resend, client);
|
2014-01-18 19:32:45 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
goto error;
|
|
|
|
|
|
2014-02-13 20:56:16 +01:00
|
|
|
|
r = sd_event_source_set_priority(client->timeout_resend,
|
|
|
|
|
client->event_priority);
|
2014-01-18 19:32:45 +01:00
|
|
|
|
if (r < 0)
|
2013-12-09 22:43:27 +01:00
|
|
|
|
goto error;
|
2013-12-09 22:43:25 +01:00
|
|
|
|
|
2013-12-09 22:43:27 +01:00
|
|
|
|
switch (client->state) {
|
|
|
|
|
case DHCP_STATE_INIT:
|
2014-03-19 14:45:35 +01:00
|
|
|
|
r = client_send_discover(client);
|
2014-01-18 19:32:45 +01:00
|
|
|
|
if (r >= 0) {
|
2013-12-09 22:43:27 +01:00
|
|
|
|
client->state = DHCP_STATE_SELECTING;
|
|
|
|
|
client->attempt = 1;
|
|
|
|
|
} else {
|
|
|
|
|
if (client->attempt >= 64)
|
|
|
|
|
goto error;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case DHCP_STATE_SELECTING:
|
2014-03-19 14:45:35 +01:00
|
|
|
|
r = client_send_discover(client);
|
2014-01-18 19:32:45 +01:00
|
|
|
|
if (r < 0 && client->attempt >= 64)
|
2013-12-09 22:43:25 +01:00
|
|
|
|
goto error;
|
|
|
|
|
|
2013-12-09 22:43:27 +01:00
|
|
|
|
break;
|
|
|
|
|
|
2014-01-31 10:31:22 +01:00
|
|
|
|
case DHCP_STATE_INIT_REBOOT:
|
2013-12-09 22:43:27 +01:00
|
|
|
|
case DHCP_STATE_REQUESTING:
|
2013-12-20 16:16:18 +01:00
|
|
|
|
case DHCP_STATE_RENEWING:
|
2013-12-20 16:16:20 +01:00
|
|
|
|
case DHCP_STATE_REBINDING:
|
2014-03-19 14:45:35 +01:00
|
|
|
|
r = client_send_request(client);
|
2014-01-18 19:32:45 +01:00
|
|
|
|
if (r < 0 && client->attempt >= 64)
|
2013-12-09 22:43:27 +01:00
|
|
|
|
goto error;
|
2013-12-09 22:43:25 +01:00
|
|
|
|
|
2014-01-31 10:31:22 +01:00
|
|
|
|
if (client->state == DHCP_STATE_INIT_REBOOT)
|
|
|
|
|
client->state = DHCP_STATE_REBOOTING;
|
|
|
|
|
|
2014-03-18 14:13:01 +01:00
|
|
|
|
client->request_sent = time_now;
|
2013-12-09 22:43:30 +01:00
|
|
|
|
|
2013-12-09 22:43:25 +01:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case DHCP_STATE_REBOOTING:
|
|
|
|
|
case DHCP_STATE_BOUND:
|
|
|
|
|
|
|
|
|
|
break;
|
2014-04-09 12:12:08 +02:00
|
|
|
|
|
|
|
|
|
case DHCP_STATE_STOPPED:
|
|
|
|
|
r = -EINVAL;
|
|
|
|
|
goto error;
|
2013-12-09 22:43:25 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
error:
|
2014-01-18 19:32:45 +01:00
|
|
|
|
client_stop(client, r);
|
2013-12-09 22:43:25 +01:00
|
|
|
|
|
|
|
|
|
/* Errors were dealt with when stopping the client, don't spill
|
|
|
|
|
errors into the event loop handler */
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-02-13 20:56:16 +01:00
|
|
|
|
static int client_initialize_events(sd_dhcp_client *client,
|
2014-03-18 14:13:01 +01:00
|
|
|
|
sd_event_io_handler_t io_callback) {
|
2013-12-20 16:16:19 +01:00
|
|
|
|
int r;
|
|
|
|
|
|
2014-01-18 19:32:45 +01:00
|
|
|
|
assert(client);
|
|
|
|
|
assert(client->event);
|
|
|
|
|
|
2014-02-19 23:54:58 +01:00
|
|
|
|
r = sd_event_add_io(client->event, &client->receive_message,
|
|
|
|
|
client->fd, EPOLLIN, io_callback,
|
|
|
|
|
client);
|
2013-12-20 16:16:19 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
goto error;
|
|
|
|
|
|
2014-02-13 20:56:16 +01:00
|
|
|
|
r = sd_event_source_set_priority(client->receive_message,
|
|
|
|
|
client->event_priority);
|
2014-01-18 19:32:45 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
goto error;
|
|
|
|
|
|
2014-02-20 21:04:03 +01:00
|
|
|
|
client->timeout_resend = sd_event_source_unref(client->timeout_resend);
|
|
|
|
|
|
2014-03-24 02:49:09 +01:00
|
|
|
|
r = sd_event_add_time(client->event,
|
|
|
|
|
&client->timeout_resend,
|
|
|
|
|
CLOCK_MONOTONIC,
|
|
|
|
|
0, 0,
|
|
|
|
|
client_timeout_resend, client);
|
2014-01-18 19:32:45 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
goto error;
|
|
|
|
|
|
2014-02-13 20:56:16 +01:00
|
|
|
|
r = sd_event_source_set_priority(client->timeout_resend,
|
|
|
|
|
client->event_priority);
|
2013-12-20 16:16:19 +01:00
|
|
|
|
|
|
|
|
|
error:
|
|
|
|
|
if (r < 0)
|
|
|
|
|
client_stop(client, r);
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2014-03-12 10:46:40 +01:00
|
|
|
|
static int client_start(sd_dhcp_client *client) {
|
|
|
|
|
int r;
|
|
|
|
|
|
|
|
|
|
assert_return(client, -EINVAL);
|
|
|
|
|
assert_return(client->event, -EINVAL);
|
|
|
|
|
assert_return(client->index > 0, -EINVAL);
|
|
|
|
|
assert_return(client->fd < 0, -EBUSY);
|
|
|
|
|
assert_return(client->xid == 0, -EINVAL);
|
|
|
|
|
assert_return(client->state == DHCP_STATE_INIT ||
|
|
|
|
|
client->state == DHCP_STATE_INIT_REBOOT, -EBUSY);
|
|
|
|
|
|
|
|
|
|
client->xid = random_u32();
|
|
|
|
|
|
2014-06-19 15:14:14 +02:00
|
|
|
|
r = dhcp_network_bind_raw_socket(client->index, &client->link, client->xid, client->client_id.mac_addr);
|
2014-03-12 10:46:40 +01:00
|
|
|
|
if (r < 0) {
|
|
|
|
|
client_stop(client, r);
|
|
|
|
|
return r;
|
|
|
|
|
}
|
|
|
|
|
client->fd = r;
|
2014-03-20 16:21:43 +01:00
|
|
|
|
|
|
|
|
|
if (client->state == DHCP_STATE_INIT) {
|
|
|
|
|
client->start_time = now(CLOCK_MONOTONIC);
|
|
|
|
|
client->secs = 0;
|
|
|
|
|
}
|
2014-03-12 10:46:40 +01:00
|
|
|
|
|
|
|
|
|
return client_initialize_events(client, client_receive_message_raw);
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-09 22:43:30 +01:00
|
|
|
|
static int client_timeout_expire(sd_event_source *s, uint64_t usec,
|
2014-01-16 19:55:25 +01:00
|
|
|
|
void *userdata) {
|
2013-12-09 22:43:31 +01:00
|
|
|
|
sd_dhcp_client *client = userdata;
|
2014-05-22 15:18:28 +02:00
|
|
|
|
DHCP_CLIENT_DONT_DESTROY(client);
|
2013-12-09 22:43:31 +01:00
|
|
|
|
|
2014-02-22 19:53:45 +01:00
|
|
|
|
log_dhcp_client(client, "EXPIRED");
|
|
|
|
|
|
2014-05-22 15:18:28 +02:00
|
|
|
|
client_notify(client, DHCP_EVENT_EXPIRED);
|
2014-03-12 10:46:40 +01:00
|
|
|
|
|
2014-04-09 12:12:08 +02:00
|
|
|
|
/* lease was lost, start over if not freed or stopped in callback */
|
2014-05-22 15:18:28 +02:00
|
|
|
|
if (client->state != DHCP_STATE_STOPPED) {
|
2014-04-09 12:12:07 +02:00
|
|
|
|
client_initialize(client);
|
|
|
|
|
client_start(client);
|
|
|
|
|
}
|
2013-12-09 22:43:31 +01:00
|
|
|
|
|
2013-12-09 22:43:30 +01:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-01-16 19:55:25 +01:00
|
|
|
|
static int client_timeout_t2(sd_event_source *s, uint64_t usec, void *userdata) {
|
2013-12-20 16:16:20 +01:00
|
|
|
|
sd_dhcp_client *client = userdata;
|
2014-05-22 15:18:28 +02:00
|
|
|
|
DHCP_CLIENT_DONT_DESTROY(client);
|
2013-12-20 16:16:20 +01:00
|
|
|
|
int r;
|
|
|
|
|
|
2014-03-18 19:22:43 +01:00
|
|
|
|
client->receive_message = sd_event_source_unref(client->receive_message);
|
2014-05-06 22:59:22 +02:00
|
|
|
|
client->fd = asynchronous_close(client->fd);
|
2013-12-20 16:16:20 +01:00
|
|
|
|
|
|
|
|
|
client->state = DHCP_STATE_REBINDING;
|
|
|
|
|
client->attempt = 1;
|
|
|
|
|
|
2014-06-19 15:14:14 +02:00
|
|
|
|
r = dhcp_network_bind_raw_socket(client->index, &client->link, client->xid, client->client_id.mac_addr);
|
2013-12-20 16:16:20 +01:00
|
|
|
|
if (r < 0) {
|
|
|
|
|
client_stop(client, r);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
client->fd = r;
|
|
|
|
|
|
2014-03-18 14:13:01 +01:00
|
|
|
|
return client_initialize_events(client, client_receive_message_raw);
|
2013-12-09 22:43:30 +01:00
|
|
|
|
}
|
|
|
|
|
|
2014-02-13 20:56:16 +01:00
|
|
|
|
static int client_timeout_t1(sd_event_source *s, uint64_t usec,
|
|
|
|
|
void *userdata) {
|
2013-12-20 16:16:18 +01:00
|
|
|
|
sd_dhcp_client *client = userdata;
|
2014-05-22 15:18:28 +02:00
|
|
|
|
DHCP_CLIENT_DONT_DESTROY(client);
|
2013-12-20 16:16:18 +01:00
|
|
|
|
int r;
|
|
|
|
|
|
|
|
|
|
client->state = DHCP_STATE_RENEWING;
|
|
|
|
|
client->attempt = 1;
|
|
|
|
|
|
2014-05-06 21:57:09 +02:00
|
|
|
|
r = dhcp_network_bind_udp_socket(client->lease->address,
|
2014-02-14 17:23:58 +01:00
|
|
|
|
DHCP_PORT_CLIENT);
|
2013-12-20 16:16:19 +01:00
|
|
|
|
if (r < 0) {
|
2014-06-29 16:42:43 +02:00
|
|
|
|
log_dhcp_client(client, "could not bind UDP socket");
|
2013-12-20 16:16:19 +01:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
2013-12-20 16:16:18 +01:00
|
|
|
|
|
|
|
|
|
client->fd = r;
|
|
|
|
|
|
2014-03-18 14:13:01 +01:00
|
|
|
|
return client_initialize_events(client, client_receive_message_udp);
|
2013-12-09 22:43:30 +01:00
|
|
|
|
}
|
|
|
|
|
|
2014-02-13 20:56:16 +01:00
|
|
|
|
static int client_handle_offer(sd_dhcp_client *client, DHCPMessage *offer,
|
|
|
|
|
size_t len) {
|
2014-02-04 23:13:52 +01:00
|
|
|
|
_cleanup_dhcp_lease_unref_ sd_dhcp_lease *lease = NULL;
|
2014-01-16 19:55:25 +01:00
|
|
|
|
int r;
|
2013-12-09 22:43:29 +01:00
|
|
|
|
|
2014-02-04 23:13:52 +01:00
|
|
|
|
r = dhcp_lease_new(&lease);
|
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
2013-12-09 22:43:26 +01:00
|
|
|
|
|
2014-02-13 20:56:16 +01:00
|
|
|
|
r = dhcp_option_parse(offer, len, dhcp_lease_parse_options, lease);
|
2014-03-19 16:05:44 +01:00
|
|
|
|
if (r != DHCP_OFFER) {
|
|
|
|
|
log_dhcp_client(client, "receieved message was not an OFFER, ignoring");
|
2014-01-16 19:55:25 +01:00
|
|
|
|
return -ENOMSG;
|
2014-03-19 16:05:44 +01:00
|
|
|
|
}
|
2013-12-09 22:43:26 +01:00
|
|
|
|
|
2014-03-03 17:13:59 +01:00
|
|
|
|
lease->next_server = offer->siaddr;
|
|
|
|
|
|
2014-02-13 20:56:16 +01:00
|
|
|
|
lease->address = offer->yiaddr;
|
2013-12-09 22:43:26 +01:00
|
|
|
|
|
|
|
|
|
if (lease->address == INADDR_ANY ||
|
|
|
|
|
lease->server_address == INADDR_ANY ||
|
2014-03-19 16:05:44 +01:00
|
|
|
|
lease->lifetime == 0) {
|
|
|
|
|
log_dhcp_client(client, "receieved lease lacks address, server "
|
|
|
|
|
"address or lease lifetime, ignoring");
|
2014-01-16 19:55:25 +01:00
|
|
|
|
return -ENOMSG;
|
2014-03-19 16:05:44 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (lease->subnet_mask == INADDR_ANY) {
|
|
|
|
|
r = dhcp_lease_set_default_subnet_mask(lease);
|
|
|
|
|
if (r < 0) {
|
|
|
|
|
log_dhcp_client(client, "receieved lease lacks subnet "
|
|
|
|
|
"mask, and a fallback one can not be "
|
|
|
|
|
"generated, ignoring");
|
|
|
|
|
return -ENOMSG;
|
|
|
|
|
}
|
|
|
|
|
}
|
2013-12-09 22:43:26 +01:00
|
|
|
|
|
2014-04-11 05:45:46 +02:00
|
|
|
|
sd_dhcp_lease_unref(client->lease);
|
2013-12-09 22:43:26 +01:00
|
|
|
|
client->lease = lease;
|
2014-01-16 19:55:25 +01:00
|
|
|
|
lease = NULL;
|
2013-12-09 22:43:26 +01:00
|
|
|
|
|
2014-02-22 19:53:45 +01:00
|
|
|
|
log_dhcp_client(client, "OFFER");
|
|
|
|
|
|
2013-12-09 22:43:26 +01:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-02-13 20:56:16 +01:00
|
|
|
|
static int client_handle_ack(sd_dhcp_client *client, DHCPMessage *ack,
|
|
|
|
|
size_t len) {
|
2014-02-04 23:13:52 +01:00
|
|
|
|
_cleanup_dhcp_lease_unref_ sd_dhcp_lease *lease = NULL;
|
2014-01-16 19:55:25 +01:00
|
|
|
|
int r;
|
2013-12-09 22:43:29 +01:00
|
|
|
|
|
2014-02-04 23:13:52 +01:00
|
|
|
|
r = dhcp_lease_new(&lease);
|
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
2013-12-09 22:43:29 +01:00
|
|
|
|
|
2014-02-13 20:56:16 +01:00
|
|
|
|
r = dhcp_option_parse(ack, len, dhcp_lease_parse_options, lease);
|
2014-02-22 19:53:45 +01:00
|
|
|
|
if (r == DHCP_NAK) {
|
|
|
|
|
log_dhcp_client(client, "NAK");
|
2014-01-16 19:55:25 +01:00
|
|
|
|
return DHCP_EVENT_NO_LEASE;
|
2014-02-22 19:53:45 +01:00
|
|
|
|
}
|
2013-12-09 22:43:29 +01:00
|
|
|
|
|
2014-03-19 16:05:44 +01:00
|
|
|
|
if (r != DHCP_ACK) {
|
|
|
|
|
log_dhcp_client(client, "receieved message was not an ACK, ignoring");
|
2014-01-16 19:55:25 +01:00
|
|
|
|
return -ENOMSG;
|
2014-03-19 16:05:44 +01:00
|
|
|
|
}
|
2013-12-09 22:43:29 +01:00
|
|
|
|
|
2014-03-03 17:13:59 +01:00
|
|
|
|
lease->next_server = ack->siaddr;
|
|
|
|
|
|
2014-02-13 20:56:16 +01:00
|
|
|
|
lease->address = ack->yiaddr;
|
2013-12-09 22:43:29 +01:00
|
|
|
|
|
|
|
|
|
if (lease->address == INADDR_ANY ||
|
|
|
|
|
lease->server_address == INADDR_ANY ||
|
2014-03-19 16:05:44 +01:00
|
|
|
|
lease->lifetime == 0) {
|
|
|
|
|
log_dhcp_client(client, "receieved lease lacks address, server "
|
|
|
|
|
"address or lease lifetime, ignoring");
|
2014-01-16 19:55:25 +01:00
|
|
|
|
return -ENOMSG;
|
2014-03-19 16:05:44 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (lease->subnet_mask == INADDR_ANY) {
|
|
|
|
|
r = dhcp_lease_set_default_subnet_mask(lease);
|
|
|
|
|
if (r < 0) {
|
|
|
|
|
log_dhcp_client(client, "receieved lease lacks subnet "
|
|
|
|
|
"mask, and a fallback one can not be "
|
|
|
|
|
"generated, ignoring");
|
|
|
|
|
return -ENOMSG;
|
|
|
|
|
}
|
|
|
|
|
}
|
2013-12-09 22:43:29 +01:00
|
|
|
|
|
|
|
|
|
r = DHCP_EVENT_IP_ACQUIRE;
|
|
|
|
|
if (client->lease) {
|
|
|
|
|
if (client->lease->address != lease->address ||
|
|
|
|
|
client->lease->subnet_mask != lease->subnet_mask ||
|
|
|
|
|
client->lease->router != lease->router) {
|
|
|
|
|
r = DHCP_EVENT_IP_CHANGE;
|
2014-06-26 15:18:43 +02:00
|
|
|
|
} else
|
|
|
|
|
r = DHCP_EVENT_RENEW;
|
2013-12-09 22:43:29 +01:00
|
|
|
|
|
2014-02-04 23:13:52 +01:00
|
|
|
|
client->lease = sd_dhcp_lease_unref(client->lease);
|
2013-12-09 22:43:29 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
client->lease = lease;
|
2014-01-16 19:55:25 +01:00
|
|
|
|
lease = NULL;
|
2013-12-09 22:43:29 +01:00
|
|
|
|
|
2014-02-22 19:53:45 +01:00
|
|
|
|
log_dhcp_client(client, "ACK");
|
|
|
|
|
|
2013-12-09 22:43:29 +01:00
|
|
|
|
return r;
|
|
|
|
|
}
|
|
|
|
|
|
2014-03-19 17:19:22 +01:00
|
|
|
|
static uint64_t client_compute_timeout(sd_dhcp_client *client,
|
|
|
|
|
uint32_t lifetime, double factor) {
|
|
|
|
|
assert(client);
|
|
|
|
|
assert(client->request_sent);
|
|
|
|
|
assert(lifetime);
|
|
|
|
|
|
|
|
|
|
return client->request_sent + ((lifetime - 3) * USEC_PER_SEC * factor) +
|
2013-12-22 19:59:12 +01:00
|
|
|
|
+ (random_u32() & 0x1fffff);
|
2013-12-09 22:43:30 +01:00
|
|
|
|
}
|
|
|
|
|
|
2014-03-19 17:19:22 +01:00
|
|
|
|
static int client_set_lease_timeouts(sd_dhcp_client *client) {
|
|
|
|
|
usec_t time_now;
|
|
|
|
|
uint64_t lifetime_timeout;
|
|
|
|
|
uint64_t t2_timeout;
|
|
|
|
|
uint64_t t1_timeout;
|
|
|
|
|
char time_string[FORMAT_TIMESPAN_MAX];
|
2014-01-16 19:55:25 +01:00
|
|
|
|
int r;
|
2013-12-09 22:43:30 +01:00
|
|
|
|
|
2014-01-18 19:32:45 +01:00
|
|
|
|
assert(client);
|
|
|
|
|
assert(client->event);
|
2014-03-19 17:19:22 +01:00
|
|
|
|
assert(client->lease);
|
|
|
|
|
assert(client->lease->lifetime);
|
2013-12-09 22:43:30 +01:00
|
|
|
|
|
2013-12-20 16:16:18 +01:00
|
|
|
|
client->timeout_t1 = sd_event_source_unref(client->timeout_t1);
|
|
|
|
|
client->timeout_t2 = sd_event_source_unref(client->timeout_t2);
|
|
|
|
|
client->timeout_expire = sd_event_source_unref(client->timeout_expire);
|
|
|
|
|
|
2014-03-19 17:19:22 +01:00
|
|
|
|
/* don't set timers for infinite leases */
|
|
|
|
|
if (client->lease->lifetime == 0xffffffff)
|
|
|
|
|
return 0;
|
2013-12-09 22:43:30 +01:00
|
|
|
|
|
2014-03-24 02:49:09 +01:00
|
|
|
|
r = sd_event_now(client->event, CLOCK_MONOTONIC, &time_now);
|
2014-03-19 17:19:22 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
|
|
|
|
assert(client->request_sent <= time_now);
|
|
|
|
|
|
|
|
|
|
/* convert the various timeouts from relative (secs) to absolute (usecs) */
|
|
|
|
|
lifetime_timeout = client_compute_timeout(client, client->lease->lifetime, 1);
|
|
|
|
|
if (client->lease->t1 && client->lease->t2) {
|
|
|
|
|
/* both T1 and T2 are given */
|
|
|
|
|
if (client->lease->t1 < client->lease->t2 &&
|
|
|
|
|
client->lease->t2 < client->lease->lifetime) {
|
|
|
|
|
/* they are both valid */
|
|
|
|
|
t2_timeout = client_compute_timeout(client, client->lease->t2, 1);
|
|
|
|
|
t1_timeout = client_compute_timeout(client, client->lease->t1, 1);
|
|
|
|
|
} else {
|
|
|
|
|
/* discard both */
|
|
|
|
|
t2_timeout = client_compute_timeout(client, client->lease->lifetime, 7.0 / 8.0);
|
|
|
|
|
client->lease->t2 = (client->lease->lifetime * 7) / 8;
|
|
|
|
|
t1_timeout = client_compute_timeout(client, client->lease->lifetime, 0.5);
|
|
|
|
|
client->lease->t1 = client->lease->lifetime / 2;
|
|
|
|
|
}
|
|
|
|
|
} else if (client->lease->t2 && client->lease->t2 < client->lease->lifetime) {
|
|
|
|
|
/* only T2 is given, and it is valid */
|
|
|
|
|
t2_timeout = client_compute_timeout(client, client->lease->t2, 1);
|
|
|
|
|
t1_timeout = client_compute_timeout(client, client->lease->lifetime, 0.5);
|
|
|
|
|
client->lease->t1 = client->lease->lifetime / 2;
|
|
|
|
|
if (t2_timeout <= t1_timeout) {
|
|
|
|
|
/* the computed T1 would be invalid, so discard T2 */
|
|
|
|
|
t2_timeout = client_compute_timeout(client, client->lease->lifetime, 7.0 / 8.0);
|
|
|
|
|
client->lease->t2 = (client->lease->lifetime * 7) / 8;
|
|
|
|
|
}
|
|
|
|
|
} else if (client->lease->t1 && client->lease->t1 < client->lease->lifetime) {
|
|
|
|
|
/* only T1 is given, and it is valid */
|
|
|
|
|
t1_timeout = client_compute_timeout(client, client->lease->t1, 1);
|
|
|
|
|
t2_timeout = client_compute_timeout(client, client->lease->lifetime, 7.0 / 8.0);
|
|
|
|
|
client->lease->t2 = (client->lease->lifetime * 7) / 8;
|
|
|
|
|
if (t2_timeout <= t1_timeout) {
|
|
|
|
|
/* the computed T2 would be invalid, so discard T1 */
|
|
|
|
|
t2_timeout = client_compute_timeout(client, client->lease->lifetime, 0.5);
|
|
|
|
|
client->lease->t2 = client->lease->lifetime / 2;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
/* fall back to the default timeouts */
|
|
|
|
|
t1_timeout = client_compute_timeout(client, client->lease->lifetime, 0.5);
|
|
|
|
|
client->lease->t1 = client->lease->lifetime / 2;
|
|
|
|
|
t2_timeout = client_compute_timeout(client, client->lease->lifetime, 7.0 / 8.0);
|
|
|
|
|
client->lease->t2 = (client->lease->lifetime * 7) / 8;
|
|
|
|
|
}
|
2013-12-09 22:43:30 +01:00
|
|
|
|
|
2014-03-19 17:19:22 +01:00
|
|
|
|
/* arm lifetime timeout */
|
2014-03-24 02:49:09 +01:00
|
|
|
|
r = sd_event_add_time(client->event, &client->timeout_expire,
|
|
|
|
|
CLOCK_MONOTONIC,
|
|
|
|
|
lifetime_timeout, 10 * USEC_PER_MSEC,
|
|
|
|
|
client_timeout_expire, client);
|
2014-01-16 19:55:25 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
2013-12-09 22:43:30 +01:00
|
|
|
|
|
2014-03-19 17:19:22 +01:00
|
|
|
|
r = sd_event_source_set_priority(client->timeout_expire,
|
2014-02-13 20:56:16 +01:00
|
|
|
|
client->event_priority);
|
2014-01-18 19:32:45 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
|
|
|
|
|
2014-03-19 17:19:22 +01:00
|
|
|
|
log_dhcp_client(client, "lease expires in %s",
|
|
|
|
|
format_timespan(time_string, FORMAT_TIMESPAN_MAX,
|
|
|
|
|
lifetime_timeout - time_now, 0));
|
2013-12-09 22:43:30 +01:00
|
|
|
|
|
2014-03-19 17:19:22 +01:00
|
|
|
|
/* don't arm earlier timeouts if this has already expired */
|
|
|
|
|
if (lifetime_timeout <= time_now)
|
|
|
|
|
return 0;
|
2013-12-09 22:43:30 +01:00
|
|
|
|
|
2014-03-19 17:19:22 +01:00
|
|
|
|
/* arm T2 timeout */
|
2014-03-24 02:49:09 +01:00
|
|
|
|
r = sd_event_add_time(client->event,
|
|
|
|
|
&client->timeout_t2,
|
|
|
|
|
CLOCK_MONOTONIC,
|
|
|
|
|
t2_timeout,
|
|
|
|
|
10 * USEC_PER_MSEC,
|
|
|
|
|
client_timeout_t2, client);
|
2014-01-16 19:55:25 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
2013-12-09 22:43:30 +01:00
|
|
|
|
|
2014-02-13 20:56:16 +01:00
|
|
|
|
r = sd_event_source_set_priority(client->timeout_t2,
|
|
|
|
|
client->event_priority);
|
2014-01-18 19:32:45 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
|
|
|
|
|
2014-03-19 17:19:22 +01:00
|
|
|
|
log_dhcp_client(client, "T2 expires in %s",
|
|
|
|
|
format_timespan(time_string, FORMAT_TIMESPAN_MAX,
|
|
|
|
|
t2_timeout - time_now, 0));
|
|
|
|
|
|
|
|
|
|
/* don't arm earlier timeout if this has already expired */
|
|
|
|
|
if (t2_timeout <= time_now)
|
|
|
|
|
return 0;
|
2013-12-09 22:43:30 +01:00
|
|
|
|
|
2014-03-19 17:19:22 +01:00
|
|
|
|
/* arm T1 timeout */
|
2014-03-24 02:49:09 +01:00
|
|
|
|
r = sd_event_add_time(client->event,
|
|
|
|
|
&client->timeout_t1,
|
|
|
|
|
CLOCK_MONOTONIC,
|
|
|
|
|
t1_timeout, 10 * USEC_PER_MSEC,
|
|
|
|
|
client_timeout_t1, client);
|
2014-01-16 19:55:25 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
2013-12-09 22:43:30 +01:00
|
|
|
|
|
2014-03-19 17:19:22 +01:00
|
|
|
|
r = sd_event_source_set_priority(client->timeout_t1,
|
2014-02-13 20:56:16 +01:00
|
|
|
|
client->event_priority);
|
2014-01-18 19:32:45 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
|
|
|
|
|
2014-03-19 17:19:22 +01:00
|
|
|
|
log_dhcp_client(client, "T1 expires in %s",
|
|
|
|
|
format_timespan(time_string, FORMAT_TIMESPAN_MAX,
|
|
|
|
|
t1_timeout - time_now, 0));
|
|
|
|
|
|
2013-12-09 22:43:30 +01:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-02-13 20:56:16 +01:00
|
|
|
|
static int client_handle_message(sd_dhcp_client *client, DHCPMessage *message,
|
2014-03-19 17:19:22 +01:00
|
|
|
|
int len) {
|
2014-05-22 15:18:28 +02:00
|
|
|
|
DHCP_CLIENT_DONT_DESTROY(client);
|
2014-02-13 20:56:16 +01:00
|
|
|
|
int r = 0, notify_event = 0;
|
2013-12-09 22:43:26 +01:00
|
|
|
|
|
2014-01-18 19:32:45 +01:00
|
|
|
|
assert(client);
|
|
|
|
|
assert(client->event);
|
2014-02-13 20:56:16 +01:00
|
|
|
|
assert(message);
|
2014-01-18 19:32:45 +01:00
|
|
|
|
|
2014-04-06 14:05:32 +02:00
|
|
|
|
if (be32toh(message->magic) != DHCP_MAGIC_COOKIE) {
|
|
|
|
|
log_dhcp_client(client, "not a DHCP message: ignoring");
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-02-23 14:15:05 +01:00
|
|
|
|
if (message->op != BOOTREPLY) {
|
|
|
|
|
log_dhcp_client(client, "not a BOOTREPLY message: ignoring");
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (be32toh(message->xid) != client->xid) {
|
|
|
|
|
log_dhcp_client(client, "received xid (%u) does not match "
|
|
|
|
|
"expected (%u): ignoring",
|
|
|
|
|
be32toh(message->xid), client->xid);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2013-12-09 22:43:27 +01:00
|
|
|
|
|
2014-04-06 14:05:32 +02:00
|
|
|
|
if (message->htype != ARPHRD_ETHER || message->hlen != ETHER_ADDR_LEN) {
|
|
|
|
|
log_dhcp_client(client, "not an ethernet packet");
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-03-19 12:53:02 +01:00
|
|
|
|
if (memcmp(&message->chaddr[0], &client->client_id.mac_addr,
|
|
|
|
|
ETH_ALEN)) {
|
2014-02-23 14:15:05 +01:00
|
|
|
|
log_dhcp_client(client, "received chaddr does not match "
|
|
|
|
|
"expected: ignoring");
|
2014-02-16 23:28:19 +01:00
|
|
|
|
return 0;
|
2014-02-23 14:15:05 +01:00
|
|
|
|
}
|
2013-12-09 22:43:26 +01:00
|
|
|
|
|
|
|
|
|
switch (client->state) {
|
|
|
|
|
case DHCP_STATE_SELECTING:
|
|
|
|
|
|
2014-02-13 20:56:16 +01:00
|
|
|
|
r = client_handle_offer(client, message, len);
|
|
|
|
|
if (r >= 0) {
|
2013-12-09 22:43:26 +01:00
|
|
|
|
|
|
|
|
|
client->timeout_resend =
|
|
|
|
|
sd_event_source_unref(client->timeout_resend);
|
|
|
|
|
|
|
|
|
|
client->state = DHCP_STATE_REQUESTING;
|
2013-12-09 22:43:27 +01:00
|
|
|
|
client->attempt = 1;
|
|
|
|
|
|
2014-03-24 02:49:09 +01:00
|
|
|
|
r = sd_event_add_time(client->event,
|
|
|
|
|
&client->timeout_resend,
|
|
|
|
|
CLOCK_MONOTONIC,
|
|
|
|
|
0, 0,
|
|
|
|
|
client_timeout_resend, client);
|
2013-12-09 22:43:29 +01:00
|
|
|
|
if (r < 0)
|
2013-12-09 22:43:27 +01:00
|
|
|
|
goto error;
|
2014-01-18 19:32:45 +01:00
|
|
|
|
|
2014-02-13 20:56:16 +01:00
|
|
|
|
r = sd_event_source_set_priority(client->timeout_resend,
|
|
|
|
|
client->event_priority);
|
2014-01-18 19:32:45 +01:00
|
|
|
|
if (r < 0)
|
|
|
|
|
goto error;
|
2014-03-19 16:05:44 +01:00
|
|
|
|
} else if (r == -ENOMSG)
|
|
|
|
|
/* invalid message, let's ignore it */
|
|
|
|
|
return 0;
|
2013-12-09 22:43:26 +01:00
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
2014-01-31 10:31:22 +01:00
|
|
|
|
case DHCP_STATE_REBOOTING:
|
2013-12-09 22:43:29 +01:00
|
|
|
|
case DHCP_STATE_REQUESTING:
|
2013-12-20 16:16:18 +01:00
|
|
|
|
case DHCP_STATE_RENEWING:
|
2013-12-20 16:16:20 +01:00
|
|
|
|
case DHCP_STATE_REBINDING:
|
2013-12-20 16:16:18 +01:00
|
|
|
|
|
2014-02-13 20:56:16 +01:00
|
|
|
|
r = client_handle_ack(client, message, len);
|
2014-01-31 10:31:22 +01:00
|
|
|
|
if (r == DHCP_EVENT_NO_LEASE) {
|
|
|
|
|
|
|
|
|
|
client->timeout_resend =
|
|
|
|
|
sd_event_source_unref(client->timeout_resend);
|
|
|
|
|
|
|
|
|
|
if (client->state == DHCP_STATE_REBOOTING) {
|
|
|
|
|
r = client_initialize(client);
|
|
|
|
|
if (r < 0)
|
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
|
|
r = client_start(client);
|
|
|
|
|
if (r < 0)
|
|
|
|
|
goto error;
|
2014-04-12 01:01:13 +02:00
|
|
|
|
|
|
|
|
|
log_dhcp_client(client, "REBOOTED");
|
2014-01-31 10:31:22 +01:00
|
|
|
|
}
|
|
|
|
|
|
2013-12-09 22:43:29 +01:00
|
|
|
|
goto error;
|
2014-03-19 16:05:44 +01:00
|
|
|
|
} else if (r >= 0) {
|
2013-12-09 22:43:29 +01:00
|
|
|
|
client->timeout_resend =
|
|
|
|
|
sd_event_source_unref(client->timeout_resend);
|
|
|
|
|
|
2014-01-31 10:31:22 +01:00
|
|
|
|
if (IN_SET(client->state, DHCP_STATE_REQUESTING,
|
|
|
|
|
DHCP_STATE_REBOOTING))
|
2013-12-20 16:16:18 +01:00
|
|
|
|
notify_event = DHCP_EVENT_IP_ACQUIRE;
|
|
|
|
|
else if (r != DHCP_EVENT_IP_ACQUIRE)
|
|
|
|
|
notify_event = r;
|
|
|
|
|
|
2013-12-09 22:43:29 +01:00
|
|
|
|
client->state = DHCP_STATE_BOUND;
|
|
|
|
|
client->attempt = 1;
|
|
|
|
|
|
|
|
|
|
client->last_addr = client->lease->address;
|
|
|
|
|
|
2014-03-19 17:19:22 +01:00
|
|
|
|
r = client_set_lease_timeouts(client);
|
2013-12-20 16:16:18 +01:00
|
|
|
|
if (r < 0)
|
2013-12-09 22:43:30 +01:00
|
|
|
|
goto error;
|
|
|
|
|
|
2014-04-09 12:12:07 +02:00
|
|
|
|
if (notify_event) {
|
2014-05-22 15:18:28 +02:00
|
|
|
|
client_notify(client, notify_event);
|
|
|
|
|
if (client->state == DHCP_STATE_STOPPED)
|
2014-04-09 12:12:07 +02:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
2013-12-09 22:43:29 +01:00
|
|
|
|
|
|
|
|
|
client->receive_message =
|
|
|
|
|
sd_event_source_unref(client->receive_message);
|
2014-05-06 22:59:22 +02:00
|
|
|
|
client->fd = asynchronous_close(client->fd);
|
2014-03-19 16:05:44 +01:00
|
|
|
|
} else if (r == -ENOMSG)
|
|
|
|
|
/* invalid message, let's ignore it */
|
|
|
|
|
return 0;
|
2013-12-17 16:24:16 +01:00
|
|
|
|
|
2013-12-09 22:43:29 +01:00
|
|
|
|
break;
|
|
|
|
|
|
2013-12-09 22:43:26 +01:00
|
|
|
|
case DHCP_STATE_INIT:
|
|
|
|
|
case DHCP_STATE_INIT_REBOOT:
|
|
|
|
|
case DHCP_STATE_BOUND:
|
|
|
|
|
|
|
|
|
|
break;
|
2014-04-09 12:12:08 +02:00
|
|
|
|
|
|
|
|
|
case DHCP_STATE_STOPPED:
|
|
|
|
|
r = -EINVAL;
|
|
|
|
|
goto error;
|
2013-12-09 22:43:26 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
error:
|
2013-12-17 16:24:16 +01:00
|
|
|
|
if (r < 0 || r == DHCP_EVENT_NO_LEASE)
|
2014-04-09 12:12:07 +02:00
|
|
|
|
client_stop(client, r);
|
2013-12-09 22:43:27 +01:00
|
|
|
|
|
2014-04-09 12:12:07 +02:00
|
|
|
|
return r;
|
2013-12-09 22:43:26 +01:00
|
|
|
|
}
|
|
|
|
|
|
2014-02-19 19:44:06 +01:00
|
|
|
|
static int client_receive_message_udp(sd_event_source *s, int fd,
|
2014-02-13 20:56:16 +01:00
|
|
|
|
uint32_t revents, void *userdata) {
|
|
|
|
|
sd_dhcp_client *client = userdata;
|
2014-02-23 17:30:13 +01:00
|
|
|
|
_cleanup_free_ DHCPMessage *message = NULL;
|
|
|
|
|
int buflen = 0, len, r;
|
2014-02-13 20:56:16 +01:00
|
|
|
|
|
|
|
|
|
assert(s);
|
|
|
|
|
assert(client);
|
|
|
|
|
|
2014-02-23 17:30:13 +01:00
|
|
|
|
r = ioctl(fd, FIONREAD, &buflen);
|
2014-05-21 16:31:28 +02:00
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
|
|
|
|
|
|
|
|
|
if (buflen < 0)
|
|
|
|
|
/* this can't be right */
|
|
|
|
|
return -EIO;
|
2014-02-23 17:30:13 +01:00
|
|
|
|
|
|
|
|
|
message = malloc0(buflen);
|
|
|
|
|
if (!message)
|
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
|
|
len = read(fd, message, buflen);
|
2014-04-06 19:35:36 +02:00
|
|
|
|
if (len < 0) {
|
|
|
|
|
log_dhcp_client(client, "could not receive message from UDP "
|
2014-05-24 06:10:36 +02:00
|
|
|
|
"socket: %m");
|
2014-04-06 19:35:36 +02:00
|
|
|
|
return 0;
|
|
|
|
|
} else if ((size_t)len < sizeof(DHCPMessage))
|
2014-02-13 20:56:16 +01:00
|
|
|
|
return 0;
|
|
|
|
|
|
2014-03-19 17:19:22 +01:00
|
|
|
|
return client_handle_message(client, message, len);
|
2014-02-13 20:56:16 +01:00
|
|
|
|
}
|
|
|
|
|
|
2014-02-19 19:44:06 +01:00
|
|
|
|
static int client_receive_message_raw(sd_event_source *s, int fd,
|
2014-02-13 20:56:16 +01:00
|
|
|
|
uint32_t revents, void *userdata) {
|
|
|
|
|
sd_dhcp_client *client = userdata;
|
2014-02-23 17:30:13 +01:00
|
|
|
|
_cleanup_free_ DHCPPacket *packet = NULL;
|
2014-02-24 01:09:21 +01:00
|
|
|
|
uint8_t cmsgbuf[CMSG_LEN(sizeof(struct tpacket_auxdata))];
|
|
|
|
|
struct iovec iov = {};
|
|
|
|
|
struct msghdr msg = {
|
|
|
|
|
.msg_iov = &iov,
|
|
|
|
|
.msg_iovlen = 1,
|
|
|
|
|
.msg_control = cmsgbuf,
|
|
|
|
|
.msg_controllen = sizeof(cmsgbuf),
|
|
|
|
|
};
|
|
|
|
|
struct cmsghdr *cmsg;
|
|
|
|
|
bool checksum = true;
|
|
|
|
|
int buflen = 0, len, r;
|
2014-02-13 20:56:16 +01:00
|
|
|
|
|
|
|
|
|
assert(s);
|
|
|
|
|
assert(client);
|
|
|
|
|
|
2014-02-23 17:30:13 +01:00
|
|
|
|
r = ioctl(fd, FIONREAD, &buflen);
|
2014-05-21 16:31:28 +02:00
|
|
|
|
if (r < 0)
|
|
|
|
|
return r;
|
|
|
|
|
|
|
|
|
|
if (buflen < 0)
|
|
|
|
|
/* this can't be right */
|
|
|
|
|
return -EIO;
|
2014-02-23 17:30:13 +01:00
|
|
|
|
|
|
|
|
|
packet = malloc0(buflen);
|
|
|
|
|
if (!packet)
|
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
2014-02-24 01:09:21 +01:00
|
|
|
|
iov.iov_base = packet;
|
|
|
|
|
iov.iov_len = buflen;
|
|
|
|
|
|
|
|
|
|
len = recvmsg(fd, &msg, 0);
|
|
|
|
|
if (len < 0) {
|
|
|
|
|
log_dhcp_client(client, "could not receive message from raw "
|
2014-05-24 06:10:36 +02:00
|
|
|
|
"socket: %m");
|
2014-02-13 20:56:16 +01:00
|
|
|
|
return 0;
|
2014-04-06 19:35:36 +02:00
|
|
|
|
} else if ((size_t)len < sizeof(DHCPPacket))
|
|
|
|
|
return 0;
|
2014-02-24 01:09:21 +01:00
|
|
|
|
|
|
|
|
|
for (cmsg = CMSG_FIRSTHDR(&msg); cmsg; cmsg = CMSG_NXTHDR(&msg, cmsg)) {
|
2014-04-11 00:51:55 +02:00
|
|
|
|
if (cmsg->cmsg_level == SOL_PACKET &&
|
|
|
|
|
cmsg->cmsg_type == PACKET_AUXDATA &&
|
|
|
|
|
cmsg->cmsg_len == CMSG_LEN(sizeof(struct tpacket_auxdata))) {
|
|
|
|
|
struct tpacket_auxdata *aux = (struct tpacket_auxdata*)CMSG_DATA(cmsg);
|
2014-02-24 01:09:21 +01:00
|
|
|
|
|
|
|
|
|
checksum = !(aux->tp_status & TP_STATUS_CSUMNOTREADY);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2014-02-13 20:56:16 +01:00
|
|
|
|
|
2014-02-24 01:09:21 +01:00
|
|
|
|
r = dhcp_packet_verify_headers(packet, len, checksum);
|
2014-02-23 01:34:05 +01:00
|
|
|
|
if (r < 0)
|
2014-02-16 23:28:19 +01:00
|
|
|
|
return 0;
|
2014-02-13 20:56:16 +01:00
|
|
|
|
|
|
|
|
|
len -= DHCP_IP_UDP_SIZE;
|
|
|
|
|
|
2014-03-19 17:19:22 +01:00
|
|
|
|
return client_handle_message(client, &packet->dhcp, len);
|
2014-02-13 20:56:16 +01:00
|
|
|
|
}
|
|
|
|
|
|
2014-01-16 19:55:25 +01:00
|
|
|
|
int sd_dhcp_client_start(sd_dhcp_client *client) {
|
2013-12-20 16:16:19 +01:00
|
|
|
|
int r;
|
2013-12-09 22:43:25 +01:00
|
|
|
|
|
2013-12-09 22:43:19 +01:00
|
|
|
|
assert_return(client, -EINVAL);
|
|
|
|
|
|
2014-03-12 10:46:40 +01:00
|
|
|
|
r = client_initialize(client);
|
|
|
|
|
if (r < 0)
|
2013-12-20 16:16:19 +01:00
|
|
|
|
return r;
|
2013-12-09 22:43:26 +01:00
|
|
|
|
|
2014-03-12 10:46:40 +01:00
|
|
|
|
if (client->last_addr)
|
|
|
|
|
client->state = DHCP_STATE_INIT_REBOOT;
|
2013-12-09 22:43:25 +01:00
|
|
|
|
|
2014-04-12 01:01:13 +02:00
|
|
|
|
r = client_start(client);
|
|
|
|
|
if (r >= 0)
|
2014-04-20 19:10:02 +02:00
|
|
|
|
log_dhcp_client(client, "STARTED on ifindex %u with address %s",
|
|
|
|
|
client->index,
|
|
|
|
|
ether_ntoa(&client->client_id.mac_addr));
|
2014-04-12 01:01:13 +02:00
|
|
|
|
|
|
|
|
|
return r;
|
2013-12-09 22:43:19 +01:00
|
|
|
|
}
|
|
|
|
|
|
2014-01-16 19:55:25 +01:00
|
|
|
|
int sd_dhcp_client_stop(sd_dhcp_client *client) {
|
2014-05-22 15:18:28 +02:00
|
|
|
|
DHCP_CLIENT_DONT_DESTROY(client);
|
|
|
|
|
|
2014-04-09 12:12:07 +02:00
|
|
|
|
assert_return(client, -EINVAL);
|
|
|
|
|
|
2014-05-22 15:18:28 +02:00
|
|
|
|
client_stop(client, DHCP_EVENT_STOP);
|
|
|
|
|
client->state = DHCP_STATE_STOPPED;
|
2014-04-09 12:12:07 +02:00
|
|
|
|
|
|
|
|
|
return 0;
|
2013-12-09 22:43:23 +01:00
|
|
|
|
}
|
|
|
|
|
|
2014-02-13 20:56:16 +01:00
|
|
|
|
int sd_dhcp_client_attach_event(sd_dhcp_client *client, sd_event *event,
|
|
|
|
|
int priority) {
|
2014-01-18 19:32:45 +01:00
|
|
|
|
int r;
|
|
|
|
|
|
|
|
|
|
assert_return(client, -EINVAL);
|
|
|
|
|
assert_return(!client->event, -EBUSY);
|
|
|
|
|
|
|
|
|
|
if (event)
|
|
|
|
|
client->event = sd_event_ref(event);
|
|
|
|
|
else {
|
|
|
|
|
r = sd_event_default(&client->event);
|
|
|
|
|
if (r < 0)
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
client->event_priority = priority;
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int sd_dhcp_client_detach_event(sd_dhcp_client *client) {
|
|
|
|
|
assert_return(client, -EINVAL);
|
|
|
|
|
|
|
|
|
|
client->event = sd_event_unref(client->event);
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
sd_event *sd_dhcp_client_get_event(sd_dhcp_client *client) {
|
|
|
|
|
if (!client)
|
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
return client->event;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-09 12:12:07 +02:00
|
|
|
|
sd_dhcp_client *sd_dhcp_client_ref(sd_dhcp_client *client) {
|
|
|
|
|
if (client)
|
|
|
|
|
assert_se(REFCNT_INC(client->n_ref) >= 2);
|
|
|
|
|
|
|
|
|
|
return client;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
sd_dhcp_client *sd_dhcp_client_unref(sd_dhcp_client *client) {
|
|
|
|
|
if (client && REFCNT_DEC(client->n_ref) <= 0) {
|
2014-05-22 15:18:28 +02:00
|
|
|
|
log_dhcp_client(client, "FREE");
|
2013-12-09 22:43:32 +01:00
|
|
|
|
|
2014-04-09 12:12:07 +02:00
|
|
|
|
client_initialize(client);
|
|
|
|
|
|
|
|
|
|
client->receive_message =
|
|
|
|
|
sd_event_source_unref(client->receive_message);
|
|
|
|
|
|
|
|
|
|
sd_dhcp_client_detach_event(client);
|
|
|
|
|
|
2014-04-11 05:45:46 +02:00
|
|
|
|
sd_dhcp_lease_unref(client->lease);
|
|
|
|
|
|
2014-04-09 12:12:07 +02:00
|
|
|
|
free(client->req_opts);
|
2014-07-01 20:58:49 +02:00
|
|
|
|
free(client->hostname);
|
2014-07-14 10:04:18 +02:00
|
|
|
|
free(client->vendor_class_identifier);
|
2014-04-09 12:12:07 +02:00
|
|
|
|
free(client);
|
|
|
|
|
}
|
2013-12-09 22:43:32 +01:00
|
|
|
|
|
2014-05-22 15:18:28 +02:00
|
|
|
|
return NULL;
|
2013-12-09 22:43:32 +01:00
|
|
|
|
}
|
|
|
|
|
|
2014-01-18 19:32:45 +01:00
|
|
|
|
int sd_dhcp_client_new(sd_dhcp_client **ret) {
|
2014-05-22 15:18:28 +02:00
|
|
|
|
_cleanup_dhcp_client_unref_ sd_dhcp_client *client = NULL;
|
2013-12-09 22:43:08 +01:00
|
|
|
|
|
2014-01-18 19:32:45 +01:00
|
|
|
|
assert_return(ret, -EINVAL);
|
2013-12-09 22:43:25 +01:00
|
|
|
|
|
2013-12-09 22:43:08 +01:00
|
|
|
|
client = new0(sd_dhcp_client, 1);
|
|
|
|
|
if (!client)
|
2014-01-18 19:32:45 +01:00
|
|
|
|
return -ENOMEM;
|
2013-12-09 22:43:08 +01:00
|
|
|
|
|
2014-04-09 12:12:07 +02:00
|
|
|
|
client->n_ref = REFCNT_INIT;
|
2013-12-09 22:43:08 +01:00
|
|
|
|
client->state = DHCP_STATE_INIT;
|
|
|
|
|
client->index = -1;
|
2013-12-09 22:43:26 +01:00
|
|
|
|
client->fd = -1;
|
2013-12-09 22:43:27 +01:00
|
|
|
|
client->attempt = 1;
|
2013-12-09 22:43:08 +01:00
|
|
|
|
|
|
|
|
|
client->req_opts_size = ELEMENTSOF(default_req_opts);
|
|
|
|
|
|
|
|
|
|
client->req_opts = memdup(default_req_opts, client->req_opts_size);
|
2014-01-18 19:32:45 +01:00
|
|
|
|
if (!client->req_opts)
|
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
|
|
*ret = client;
|
|
|
|
|
client = NULL;
|
2013-12-09 22:43:08 +01:00
|
|
|
|
|
2014-01-18 19:32:45 +01:00
|
|
|
|
return 0;
|
2013-12-09 22:43:08 +01:00
|
|
|
|
}
|