2017-11-18 17:09:20 +01:00
|
|
|
/* SPDX-License-Identifier: LGPL-2.1+ */
|
2014-06-19 14:39:12 +02:00
|
|
|
/***
|
2018-06-12 17:15:23 +02:00
|
|
|
Copyright © 2014 Intel Corporation. All rights reserved.
|
2014-06-19 14:39:12 +02:00
|
|
|
***/
|
|
|
|
|
2015-10-24 22:58:24 +02:00
|
|
|
#include <net/ethernet.h>
|
2019-07-24 11:22:43 +02:00
|
|
|
#include <net/if_arp.h>
|
2014-06-19 14:39:12 +02:00
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stdio.h>
|
2015-10-24 22:58:24 +02:00
|
|
|
#include <sys/types.h>
|
2014-06-19 14:39:30 +02:00
|
|
|
#include <unistd.h>
|
2014-06-19 14:39:12 +02:00
|
|
|
|
2015-10-24 22:58:24 +02:00
|
|
|
#include "sd-dhcp6-client.h"
|
2014-06-19 14:39:12 +02:00
|
|
|
#include "sd-event.h"
|
|
|
|
|
2014-06-19 14:39:20 +02:00
|
|
|
#include "dhcp6-internal.h"
|
2014-06-19 14:39:46 +02:00
|
|
|
#include "dhcp6-lease-internal.h"
|
2015-10-24 22:58:24 +02:00
|
|
|
#include "dhcp6-protocol.h"
|
2015-10-25 13:14:12 +01:00
|
|
|
#include "fd-util.h"
|
2015-10-24 22:58:24 +02:00
|
|
|
#include "macro.h"
|
2019-03-13 12:02:21 +01:00
|
|
|
#include "memory-util.h"
|
2015-10-24 22:58:24 +02:00
|
|
|
#include "socket-util.h"
|
2018-09-13 14:31:13 +02:00
|
|
|
#include "tests.h"
|
2019-03-13 12:02:21 +01:00
|
|
|
#include "time-util.h"
|
2015-10-24 22:58:24 +02:00
|
|
|
#include "virt.h"
|
2014-06-19 14:39:12 +02:00
|
|
|
|
|
|
|
static struct ether_addr mac_addr = {
|
|
|
|
.ether_addr_octet = {'A', 'B', 'C', '1', '2', '3'}
|
|
|
|
};
|
|
|
|
|
2014-06-19 14:39:30 +02:00
|
|
|
static sd_event_source *hangcheck;
|
|
|
|
static int test_dhcp_fd[2];
|
|
|
|
static int test_index = 42;
|
2014-06-19 14:39:54 +02:00
|
|
|
static int test_client_message_num;
|
|
|
|
static be32_t test_iaid = 0;
|
|
|
|
static uint8_t test_duid[14] = { };
|
2014-06-19 14:39:30 +02:00
|
|
|
|
2014-06-19 14:39:12 +02:00
|
|
|
static int test_client_basic(sd_event *e) {
|
|
|
|
sd_dhcp6_client *client;
|
2018-09-07 17:24:15 +02:00
|
|
|
int v;
|
2014-06-19 14:39:12 +02:00
|
|
|
|
2018-10-18 21:59:18 +02:00
|
|
|
log_debug("/* %s */", __func__);
|
2014-06-19 14:39:12 +02:00
|
|
|
|
|
|
|
assert_se(sd_dhcp6_client_new(&client) >= 0);
|
|
|
|
assert_se(client);
|
|
|
|
|
|
|
|
assert_se(sd_dhcp6_client_attach_event(client, e, 0) >= 0);
|
|
|
|
|
2016-05-23 16:13:18 +02:00
|
|
|
assert_se(sd_dhcp6_client_set_ifindex(client, 15) == 0);
|
|
|
|
assert_se(sd_dhcp6_client_set_ifindex(client, -42) == -EINVAL);
|
|
|
|
assert_se(sd_dhcp6_client_set_ifindex(client, -1) == 0);
|
|
|
|
assert_se(sd_dhcp6_client_set_ifindex(client, 42) >= 0);
|
2014-06-19 14:39:12 +02:00
|
|
|
|
2014-10-08 21:15:45 +02:00
|
|
|
assert_se(sd_dhcp6_client_set_mac(client, (const uint8_t *) &mac_addr,
|
|
|
|
sizeof (mac_addr),
|
|
|
|
ARPHRD_ETHER) >= 0);
|
2014-06-19 14:39:12 +02:00
|
|
|
|
2017-11-16 10:07:07 +01:00
|
|
|
assert_se(sd_dhcp6_client_set_fqdn(client, "host") == 1);
|
|
|
|
assert_se(sd_dhcp6_client_set_fqdn(client, "host.domain") == 1);
|
|
|
|
assert_se(sd_dhcp6_client_set_fqdn(client, NULL) == 1);
|
|
|
|
assert_se(sd_dhcp6_client_set_fqdn(client, "~host") == -EINVAL);
|
|
|
|
assert_se(sd_dhcp6_client_set_fqdn(client, "~host.domain") == -EINVAL);
|
|
|
|
|
2016-01-20 14:44:28 +01:00
|
|
|
assert_se(sd_dhcp6_client_set_request_option(client, SD_DHCP6_OPTION_CLIENTID) == -EINVAL);
|
|
|
|
assert_se(sd_dhcp6_client_set_request_option(client, SD_DHCP6_OPTION_DNS_SERVERS) == -EEXIST);
|
|
|
|
assert_se(sd_dhcp6_client_set_request_option(client, SD_DHCP6_OPTION_NTP_SERVER) == -EEXIST);
|
|
|
|
assert_se(sd_dhcp6_client_set_request_option(client, SD_DHCP6_OPTION_SNTP_SERVERS) == -EEXIST);
|
|
|
|
assert_se(sd_dhcp6_client_set_request_option(client, SD_DHCP6_OPTION_DOMAIN_LIST) == -EEXIST);
|
2014-06-24 15:20:32 +02:00
|
|
|
assert_se(sd_dhcp6_client_set_request_option(client, 10) == -EINVAL);
|
|
|
|
|
2018-09-07 17:24:15 +02:00
|
|
|
assert_se(sd_dhcp6_client_set_information_request(client, 1) >= 0);
|
|
|
|
v = 0;
|
|
|
|
assert_se(sd_dhcp6_client_get_information_request(client, &v) >= 0);
|
|
|
|
assert_se(v);
|
|
|
|
assert_se(sd_dhcp6_client_set_information_request(client, 0) >= 0);
|
|
|
|
v = 42;
|
|
|
|
assert_se(sd_dhcp6_client_get_information_request(client, &v) >= 0);
|
|
|
|
assert_se(v == 0);
|
|
|
|
|
|
|
|
v = 0;
|
|
|
|
assert_se(sd_dhcp6_client_get_address_request(client, &v) >= 0);
|
|
|
|
assert_se(v);
|
|
|
|
v = 0;
|
|
|
|
assert_se(sd_dhcp6_client_set_address_request(client, 1) >= 0);
|
|
|
|
assert_se(sd_dhcp6_client_get_address_request(client, &v) >= 0);
|
|
|
|
assert_se(v);
|
|
|
|
v = 42;
|
|
|
|
assert_se(sd_dhcp6_client_set_address_request(client, 1) >= 0);
|
|
|
|
assert_se(sd_dhcp6_client_get_address_request(client, &v) >= 0);
|
|
|
|
assert_se(v);
|
|
|
|
|
|
|
|
assert_se(sd_dhcp6_client_set_address_request(client, 1) >= 0);
|
|
|
|
assert_se(sd_dhcp6_client_set_prefix_delegation(client, 1) >= 0);
|
|
|
|
v = 0;
|
|
|
|
assert_se(sd_dhcp6_client_get_address_request(client, &v) >= 0);
|
|
|
|
assert_se(v);
|
|
|
|
v = 0;
|
|
|
|
assert_se(sd_dhcp6_client_get_prefix_delegation(client, &v) >= 0);
|
|
|
|
assert_se(v);
|
|
|
|
|
2014-06-19 14:39:12 +02:00
|
|
|
assert_se(sd_dhcp6_client_set_callback(client, NULL, NULL) >= 0);
|
|
|
|
|
|
|
|
assert_se(sd_dhcp6_client_detach_event(client) >= 0);
|
|
|
|
assert_se(!sd_dhcp6_client_unref(client));
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-06-19 14:39:20 +02:00
|
|
|
static int test_option(sd_event *e) {
|
|
|
|
uint8_t packet[] = {
|
|
|
|
'F', 'O', 'O',
|
2016-01-20 14:44:28 +01:00
|
|
|
0x00, SD_DHCP6_OPTION_ORO, 0x00, 0x07,
|
2014-06-19 14:39:20 +02:00
|
|
|
'A', 'B', 'C', 'D', 'E', 'F', 'G',
|
2016-01-20 14:44:28 +01:00
|
|
|
0x00, SD_DHCP6_OPTION_VENDOR_CLASS, 0x00, 0x09,
|
2014-06-19 14:39:20 +02:00
|
|
|
'1', '2', '3', '4', '5', '6', '7', '8', '9',
|
|
|
|
'B', 'A', 'R',
|
|
|
|
};
|
|
|
|
uint8_t result[] = {
|
|
|
|
'F', 'O', 'O',
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
'B', 'A', 'R',
|
|
|
|
};
|
|
|
|
uint16_t optcode;
|
|
|
|
size_t optlen;
|
|
|
|
uint8_t *optval, *buf, *out;
|
|
|
|
size_t zero = 0, pos = 3;
|
|
|
|
size_t buflen = sizeof(packet), outlen = sizeof(result);
|
|
|
|
|
2018-10-18 21:59:18 +02:00
|
|
|
log_debug("/* %s */", __func__);
|
2014-06-19 14:39:20 +02:00
|
|
|
|
|
|
|
assert_se(buflen == outlen);
|
|
|
|
|
|
|
|
assert_se(dhcp6_option_parse(&buf, &zero, &optcode, &optlen,
|
|
|
|
&optval) == -ENOMSG);
|
|
|
|
|
|
|
|
buflen -= 3;
|
|
|
|
buf = &packet[3];
|
|
|
|
outlen -= 3;
|
|
|
|
out = &result[3];
|
|
|
|
|
|
|
|
assert_se(dhcp6_option_parse(&buf, &buflen, &optcode, &optlen,
|
|
|
|
&optval) >= 0);
|
|
|
|
pos += 4 + optlen;
|
|
|
|
assert_se(buf == &packet[pos]);
|
2016-01-20 14:44:28 +01:00
|
|
|
assert_se(optcode == SD_DHCP6_OPTION_ORO);
|
2014-06-19 14:39:20 +02:00
|
|
|
assert_se(optlen == 7);
|
|
|
|
assert_se(buflen + pos == sizeof(packet));
|
|
|
|
|
|
|
|
assert_se(dhcp6_option_append(&out, &outlen, optcode, optlen,
|
|
|
|
optval) >= 0);
|
|
|
|
assert_se(out == &result[pos]);
|
|
|
|
assert_se(*out == 0x00);
|
|
|
|
|
|
|
|
assert_se(dhcp6_option_parse(&buf, &buflen, &optcode, &optlen,
|
|
|
|
&optval) >= 0);
|
|
|
|
pos += 4 + optlen;
|
|
|
|
assert_se(buf == &packet[pos]);
|
2016-01-20 14:44:28 +01:00
|
|
|
assert_se(optcode == SD_DHCP6_OPTION_VENDOR_CLASS);
|
2014-06-19 14:39:20 +02:00
|
|
|
assert_se(optlen == 9);
|
|
|
|
assert_se(buflen + pos == sizeof(packet));
|
|
|
|
|
|
|
|
assert_se(dhcp6_option_append(&out, &outlen, optcode, optlen,
|
|
|
|
optval) >= 0);
|
|
|
|
assert_se(out == &result[pos]);
|
|
|
|
assert_se(*out == 'B');
|
|
|
|
|
|
|
|
assert_se(memcmp(packet, result, sizeof(packet)) == 0);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-01-04 14:11:44 +01:00
|
|
|
static int test_option_status(sd_event *e) {
|
|
|
|
uint8_t option1[] = {
|
|
|
|
/* IA NA */
|
|
|
|
0x00, 0x03, 0x00, 0x12, 0x1a, 0x1d, 0x1a, 0x1d,
|
|
|
|
0x00, 0x01, 0x00, 0x01, 0x00, 0x02, 0x00, 0x02,
|
|
|
|
/* status option */
|
|
|
|
0x00, 0x0d, 0x00, 0x02, 0x00, 0x01,
|
|
|
|
};
|
|
|
|
static const uint8_t option2[] = {
|
|
|
|
/* IA NA */
|
|
|
|
0x00, 0x03, 0x00, 0x2e, 0x1a, 0x1d, 0x1a, 0x1d,
|
|
|
|
0x00, 0x01, 0x00, 0x01, 0x00, 0x02, 0x00, 0x02,
|
|
|
|
/* IA Addr */
|
|
|
|
0x00, 0x05, 0x00, 0x1e,
|
|
|
|
0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08,
|
|
|
|
0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f, 0x10,
|
|
|
|
0x01, 0x02, 0x03, 0x04, 0x0a, 0x0b, 0x0c, 0x0d,
|
|
|
|
/* status option */
|
|
|
|
0x00, 0x0d, 0x00, 0x02, 0x00, 0x01,
|
|
|
|
};
|
|
|
|
static const uint8_t option3[] = {
|
|
|
|
/* IA NA */
|
|
|
|
0x00, 0x03, 0x00, 0x34, 0x1a, 0x1d, 0x1a, 0x1d,
|
|
|
|
0x00, 0x01, 0x00, 0x01, 0x00, 0x02, 0x00, 0x02,
|
|
|
|
/* IA Addr */
|
|
|
|
0x00, 0x05, 0x00, 0x24,
|
|
|
|
0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08,
|
|
|
|
0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f, 0x10,
|
|
|
|
0x01, 0x02, 0x03, 0x04, 0x0a, 0x0b, 0x0c, 0x0d,
|
|
|
|
/* status option */
|
|
|
|
0x00, 0x0d, 0x00, 0x08, 0x00, 0x00, 'f', 'o',
|
|
|
|
'o', 'b', 'a', 'r',
|
|
|
|
};
|
2018-01-04 14:11:53 +01:00
|
|
|
static const uint8_t option4[] = {
|
|
|
|
/* IA PD */
|
|
|
|
0x00, 0x19, 0x00, 0x2f, 0x1a, 0x1d, 0x1a, 0x1d,
|
|
|
|
0x00, 0x01, 0x00, 0x01, 0x00, 0x02, 0x00, 0x02,
|
|
|
|
/* IA PD Prefix */
|
|
|
|
0x00, 0x1a, 0x00, 0x1f,
|
|
|
|
0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08,
|
|
|
|
0x80, 0x20, 0x01, 0x0d, 0xb8, 0xde, 0xad, 0xbe,
|
|
|
|
0xef, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00,
|
|
|
|
/* status option */
|
|
|
|
0x00, 0x0d, 0x00, 0x02, 0x00, 0x00,
|
|
|
|
};
|
|
|
|
static const uint8_t option5[] = {
|
|
|
|
/* IA PD */
|
|
|
|
0x00, 0x19, 0x00, 0x52, 0x1a, 0x1d, 0x1a, 0x1d,
|
|
|
|
0x00, 0x01, 0x00, 0x01, 0x00, 0x02, 0x00, 0x02,
|
|
|
|
/* IA PD Prefix #1 */
|
|
|
|
0x00, 0x1a, 0x00, 0x1f,
|
|
|
|
0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08,
|
|
|
|
0x80, 0x20, 0x01, 0x0d, 0xb8, 0xde, 0xad, 0xbe,
|
|
|
|
0xef, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00,
|
|
|
|
/* status option */
|
|
|
|
0x00, 0x0d, 0x00, 0x02, 0x00, 0x00,
|
|
|
|
/* IA PD Prefix #2 */
|
|
|
|
0x00, 0x1a, 0x00, 0x1f,
|
|
|
|
0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08,
|
|
|
|
0x80, 0x20, 0x01, 0x0d, 0xb8, 0xc0, 0x0l, 0xd0,
|
|
|
|
0x0d, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00,
|
|
|
|
0x00, 0x0d, 0x00, 0x02, 0x00, 0x00,
|
|
|
|
};
|
2018-01-04 14:11:44 +01:00
|
|
|
DHCP6Option *option;
|
2018-01-04 14:11:53 +01:00
|
|
|
DHCP6IA ia, pd;
|
2018-01-04 14:11:44 +01:00
|
|
|
int r = 0;
|
|
|
|
|
2018-10-18 21:59:18 +02:00
|
|
|
log_debug("/* %s */", __func__);
|
2018-01-04 14:11:44 +01:00
|
|
|
|
|
|
|
zero(ia);
|
|
|
|
option = (DHCP6Option *)option1;
|
|
|
|
assert_se(sizeof(option1) == sizeof(DHCP6Option) + be16toh(option->len));
|
|
|
|
|
|
|
|
r = dhcp6_option_parse_ia(option, &ia);
|
|
|
|
assert_se(r == -EINVAL);
|
|
|
|
assert_se(ia.addresses == NULL);
|
|
|
|
|
|
|
|
option->len = htobe16(17);
|
|
|
|
r = dhcp6_option_parse_ia(option, &ia);
|
|
|
|
assert_se(r == -ENOBUFS);
|
|
|
|
assert_se(ia.addresses == NULL);
|
|
|
|
|
|
|
|
option->len = htobe16(sizeof(DHCP6Option));
|
|
|
|
r = dhcp6_option_parse_ia(option, &ia);
|
|
|
|
assert_se(r == -ENOBUFS);
|
|
|
|
assert_se(ia.addresses == NULL);
|
|
|
|
|
|
|
|
zero(ia);
|
|
|
|
option = (DHCP6Option *)option2;
|
|
|
|
assert_se(sizeof(option2) == sizeof(DHCP6Option) + be16toh(option->len));
|
|
|
|
|
|
|
|
r = dhcp6_option_parse_ia(option, &ia);
|
|
|
|
assert_se(r >= 0);
|
|
|
|
assert_se(ia.addresses == NULL);
|
|
|
|
|
|
|
|
zero(ia);
|
|
|
|
option = (DHCP6Option *)option3;
|
|
|
|
assert_se(sizeof(option3) == sizeof(DHCP6Option) + be16toh(option->len));
|
|
|
|
|
|
|
|
r = dhcp6_option_parse_ia(option, &ia);
|
|
|
|
assert_se(r >= 0);
|
|
|
|
assert_se(ia.addresses != NULL);
|
2018-01-15 15:36:58 +01:00
|
|
|
dhcp6_lease_free_ia(&ia);
|
2018-01-04 14:11:44 +01:00
|
|
|
|
2018-01-04 14:11:53 +01:00
|
|
|
zero(pd);
|
|
|
|
option = (DHCP6Option *)option4;
|
|
|
|
assert_se(sizeof(option4) == sizeof(DHCP6Option) + be16toh(option->len));
|
|
|
|
|
|
|
|
r = dhcp6_option_parse_ia(option, &pd);
|
|
|
|
assert_se(r == 0);
|
|
|
|
assert_se(pd.addresses != NULL);
|
|
|
|
assert_se(memcmp(&pd.ia_pd.id, &option4[4], 4) == 0);
|
|
|
|
assert_se(memcmp(&pd.ia_pd.lifetime_t1, &option4[8], 4) == 0);
|
|
|
|
assert_se(memcmp(&pd.ia_pd.lifetime_t2, &option4[12], 4) == 0);
|
2018-01-15 15:36:58 +01:00
|
|
|
dhcp6_lease_free_ia(&pd);
|
2018-01-04 14:11:53 +01:00
|
|
|
|
|
|
|
zero(pd);
|
|
|
|
option = (DHCP6Option *)option5;
|
|
|
|
assert_se(sizeof(option5) == sizeof(DHCP6Option) + be16toh(option->len));
|
|
|
|
|
|
|
|
r = dhcp6_option_parse_ia(option, &pd);
|
|
|
|
assert_se(r == 0);
|
|
|
|
assert_se(pd.addresses != NULL);
|
2018-01-15 15:36:58 +01:00
|
|
|
dhcp6_lease_free_ia(&pd);
|
2018-01-04 14:11:53 +01:00
|
|
|
|
2018-01-04 14:11:44 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-06-19 14:39:46 +02:00
|
|
|
static uint8_t msg_advertise[198] = {
|
|
|
|
0x02, 0x0f, 0xb4, 0xe5, 0x00, 0x01, 0x00, 0x0e,
|
|
|
|
0x00, 0x01, 0x00, 0x01, 0x1a, 0x6b, 0xf3, 0x30,
|
|
|
|
0x3c, 0x97, 0x0e, 0xcf, 0xa3, 0x7d, 0x00, 0x03,
|
|
|
|
0x00, 0x5e, 0x0e, 0xcf, 0xa3, 0x7d, 0x00, 0x00,
|
|
|
|
0x00, 0x50, 0x00, 0x00, 0x00, 0x78, 0x00, 0x05,
|
|
|
|
0x00, 0x18, 0x20, 0x01, 0x0d, 0xb8, 0xde, 0xad,
|
|
|
|
0xbe, 0xef, 0x78, 0xee, 0x1c, 0xf3, 0x09, 0x3c,
|
|
|
|
0x55, 0xad, 0x00, 0x00, 0x00, 0x96, 0x00, 0x00,
|
|
|
|
0x00, 0xb4, 0x00, 0x0d, 0x00, 0x32, 0x00, 0x00,
|
|
|
|
0x41, 0x64, 0x64, 0x72, 0x65, 0x73, 0x73, 0x28,
|
|
|
|
0x65, 0x73, 0x29, 0x20, 0x72, 0x65, 0x6e, 0x65,
|
|
|
|
0x77, 0x65, 0x64, 0x2e, 0x20, 0x47, 0x72, 0x65,
|
|
|
|
0x65, 0x74, 0x69, 0x6e, 0x67, 0x73, 0x20, 0x66,
|
|
|
|
0x72, 0x6f, 0x6d, 0x20, 0x70, 0x6c, 0x61, 0x6e,
|
|
|
|
0x65, 0x74, 0x20, 0x45, 0x61, 0x72, 0x74, 0x68,
|
|
|
|
0x00, 0x17, 0x00, 0x10, 0x20, 0x01, 0x0d, 0xb8,
|
|
|
|
0xde, 0xad, 0xbe, 0xef, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x01, 0x00, 0x18, 0x00, 0x0b,
|
|
|
|
0x03, 0x6c, 0x61, 0x62, 0x05, 0x69, 0x6e, 0x74,
|
|
|
|
0x72, 0x61, 0x00, 0x00, 0x1f, 0x00, 0x10, 0x20,
|
|
|
|
0x01, 0x0d, 0xb8, 0xde, 0xad, 0xbe, 0xef, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00,
|
|
|
|
0x02, 0x00, 0x0e, 0x00, 0x01, 0x00, 0x01, 0x19,
|
|
|
|
0x40, 0x5c, 0x53, 0x78, 0x2b, 0xcb, 0xb3, 0x6d,
|
|
|
|
0x53, 0x00, 0x07, 0x00, 0x01, 0x00
|
|
|
|
};
|
|
|
|
|
2014-06-19 14:39:58 +02:00
|
|
|
static uint8_t msg_reply[173] = {
|
|
|
|
0x07, 0xf7, 0x4e, 0x57, 0x00, 0x02, 0x00, 0x0e,
|
|
|
|
0x00, 0x01, 0x00, 0x01, 0x19, 0x40, 0x5c, 0x53,
|
|
|
|
0x78, 0x2b, 0xcb, 0xb3, 0x6d, 0x53, 0x00, 0x01,
|
|
|
|
0x00, 0x0e, 0x00, 0x01, 0x00, 0x01, 0x1a, 0x6b,
|
|
|
|
0xf3, 0x30, 0x3c, 0x97, 0x0e, 0xcf, 0xa3, 0x7d,
|
|
|
|
0x00, 0x03, 0x00, 0x4a, 0x0e, 0xcf, 0xa3, 0x7d,
|
|
|
|
0x00, 0x00, 0x00, 0x50, 0x00, 0x00, 0x00, 0x78,
|
|
|
|
0x00, 0x05, 0x00, 0x18, 0x20, 0x01, 0x0d, 0xb8,
|
|
|
|
0xde, 0xad, 0xbe, 0xef, 0x78, 0xee, 0x1c, 0xf3,
|
|
|
|
0x09, 0x3c, 0x55, 0xad, 0x00, 0x00, 0x00, 0x96,
|
|
|
|
0x00, 0x00, 0x00, 0xb4, 0x00, 0x0d, 0x00, 0x1e,
|
|
|
|
0x00, 0x00, 0x41, 0x6c, 0x6c, 0x20, 0x61, 0x64,
|
|
|
|
0x64, 0x72, 0x65, 0x73, 0x73, 0x65, 0x73, 0x20,
|
|
|
|
0x77, 0x65, 0x72, 0x65, 0x20, 0x61, 0x73, 0x73,
|
|
|
|
0x69, 0x67, 0x6e, 0x65, 0x64, 0x2e, 0x00, 0x17,
|
|
|
|
0x00, 0x10, 0x20, 0x01, 0x0d, 0xb8, 0xde, 0xad,
|
|
|
|
0xbe, 0xef, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x01, 0x00, 0x18, 0x00, 0x0b, 0x03, 0x6c,
|
|
|
|
0x61, 0x62, 0x05, 0x69, 0x6e, 0x74, 0x72, 0x61,
|
|
|
|
0x00, 0x00, 0x1f, 0x00, 0x10, 0x20, 0x01, 0x0d,
|
|
|
|
0xb8, 0xde, 0xad, 0xbe, 0xef, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x01
|
|
|
|
};
|
|
|
|
|
2017-11-16 10:07:07 +01:00
|
|
|
static uint8_t fqdn_wire[16] = {
|
|
|
|
0x04, 'h', 'o', 's', 't', 0x03, 'l', 'a', 'b',
|
|
|
|
0x05, 'i', 'n', 't', 'r', 'a', 0x00
|
|
|
|
};
|
|
|
|
|
2014-06-19 14:39:46 +02:00
|
|
|
static int test_advertise_option(sd_event *e) {
|
tree-wide: expose "p"-suffix unref calls in public APIs to make gcc cleanup easy
GLIB has recently started to officially support the gcc cleanup
attribute in its public API, hence let's do the same for our APIs.
With this patch we'll define an xyz_unrefp() call for each public
xyz_unref() call, to make it easy to use inside a
__attribute__((cleanup())) expression. Then, all code is ported over to
make use of this.
The new calls are also documented in the man pages, with examples how to
use them (well, I only added docs where the _unref() call itself already
had docs, and the examples, only cover sd_bus_unrefp() and
sd_event_unrefp()).
This also renames sd_lldp_free() to sd_lldp_unref(), since that's how we
tend to call our destructors these days.
Note that this defines no public macro that wraps gcc's attribute and
makes it easier to use. While I think it's our duty in the library to
make our stuff easy to use, I figure it's not our duty to make gcc's own
features easy to use on its own. Most likely, client code which wants to
make use of this should define its own:
#define _cleanup_(function) __attribute__((cleanup(function)))
Or similar, to make the gcc feature easier to use.
Making this logic public has the benefit that we can remove three header
files whose only purpose was to define these functions internally.
See #2008.
2015-11-27 19:13:45 +01:00
|
|
|
_cleanup_(sd_dhcp6_lease_unrefp) sd_dhcp6_lease *lease = NULL;
|
2014-06-19 14:39:46 +02:00
|
|
|
DHCP6Message *advertise = (DHCP6Message *)msg_advertise;
|
2018-01-04 14:11:41 +01:00
|
|
|
size_t len = sizeof(msg_advertise) - sizeof(DHCP6Message), pos = 0;
|
2014-06-19 14:39:46 +02:00
|
|
|
be32_t val;
|
|
|
|
uint8_t preference = 255;
|
|
|
|
struct in6_addr addr;
|
|
|
|
uint32_t lt_pref, lt_valid;
|
|
|
|
int r;
|
2018-01-04 14:11:41 +01:00
|
|
|
uint8_t *opt;
|
2014-06-19 14:39:46 +02:00
|
|
|
bool opt_clientid = false;
|
2015-07-10 10:42:11 +02:00
|
|
|
struct in6_addr *addrs;
|
|
|
|
char **domains;
|
2014-06-19 14:39:46 +02:00
|
|
|
|
2018-10-18 21:59:18 +02:00
|
|
|
log_debug("/* %s */", __func__);
|
2014-06-19 14:39:46 +02:00
|
|
|
|
2018-01-04 14:11:41 +01:00
|
|
|
assert_se(len >= sizeof(DHCP6Message));
|
|
|
|
|
2014-06-19 14:39:46 +02:00
|
|
|
assert_se(dhcp6_lease_new(&lease) >= 0);
|
|
|
|
|
|
|
|
assert_se(advertise->type == DHCP6_ADVERTISE);
|
|
|
|
assert_se((be32toh(advertise->transaction_id) & 0x00ffffff) ==
|
|
|
|
0x0fb4e5);
|
|
|
|
|
2018-01-04 14:11:41 +01:00
|
|
|
while (pos < len) {
|
|
|
|
DHCP6Option *option = (DHCP6Option *)&advertise->options[pos];
|
|
|
|
const uint16_t optcode = be16toh(option->code);
|
|
|
|
const uint16_t optlen = be16toh(option->len);
|
|
|
|
uint8_t *optval = option->data;
|
2014-06-19 14:39:46 +02:00
|
|
|
|
|
|
|
switch(optcode) {
|
2016-01-20 14:44:28 +01:00
|
|
|
case SD_DHCP6_OPTION_CLIENTID:
|
2014-06-19 14:39:46 +02:00
|
|
|
assert_se(optlen == 14);
|
|
|
|
|
|
|
|
opt_clientid = true;
|
|
|
|
break;
|
|
|
|
|
2016-01-20 14:44:28 +01:00
|
|
|
case SD_DHCP6_OPTION_IA_NA:
|
2014-06-19 14:39:46 +02:00
|
|
|
assert_se(optlen == 94);
|
|
|
|
assert_se(!memcmp(optval, &msg_advertise[26], optlen));
|
|
|
|
|
|
|
|
val = htobe32(0x0ecfa37d);
|
|
|
|
assert_se(!memcmp(optval, &val, sizeof(val)));
|
|
|
|
|
|
|
|
val = htobe32(80);
|
|
|
|
assert_se(!memcmp(optval + 4, &val, sizeof(val)));
|
|
|
|
|
|
|
|
val = htobe32(120);
|
|
|
|
assert_se(!memcmp(optval + 8, &val, sizeof(val)));
|
|
|
|
|
2018-01-04 14:11:41 +01:00
|
|
|
assert_se(dhcp6_option_parse_ia(option, &lease->ia) >= 0);
|
2014-06-19 14:39:46 +02:00
|
|
|
|
|
|
|
break;
|
|
|
|
|
2016-01-20 14:44:28 +01:00
|
|
|
case SD_DHCP6_OPTION_SERVERID:
|
2014-06-19 14:39:46 +02:00
|
|
|
assert_se(optlen == 14);
|
|
|
|
assert_se(!memcmp(optval, &msg_advertise[179], optlen));
|
|
|
|
|
|
|
|
assert_se(dhcp6_lease_set_serverid(lease, optval,
|
|
|
|
optlen) >= 0);
|
|
|
|
break;
|
|
|
|
|
2016-01-20 14:44:28 +01:00
|
|
|
case SD_DHCP6_OPTION_PREFERENCE:
|
2014-06-19 14:39:46 +02:00
|
|
|
assert_se(optlen == 1);
|
|
|
|
assert_se(!*optval);
|
|
|
|
|
|
|
|
assert_se(dhcp6_lease_set_preference(lease,
|
|
|
|
*optval) >= 0);
|
|
|
|
break;
|
|
|
|
|
2016-01-20 14:44:28 +01:00
|
|
|
case SD_DHCP6_OPTION_ELAPSED_TIME:
|
2014-09-01 12:21:34 +02:00
|
|
|
assert_se(optlen == 2);
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
2016-01-20 14:44:28 +01:00
|
|
|
case SD_DHCP6_OPTION_DNS_SERVERS:
|
2015-07-10 10:42:11 +02:00
|
|
|
assert_se(optlen == 16);
|
|
|
|
assert_se(dhcp6_lease_set_dns(lease, optval,
|
|
|
|
optlen) >= 0);
|
|
|
|
break;
|
|
|
|
|
2016-01-20 14:44:28 +01:00
|
|
|
case SD_DHCP6_OPTION_DOMAIN_LIST:
|
2015-07-10 10:42:11 +02:00
|
|
|
assert_se(optlen == 11);
|
|
|
|
assert_se(dhcp6_lease_set_domains(lease, optval,
|
|
|
|
optlen) >= 0);
|
|
|
|
break;
|
|
|
|
|
2016-01-20 14:44:28 +01:00
|
|
|
case SD_DHCP6_OPTION_SNTP_SERVERS:
|
2015-07-10 10:42:11 +02:00
|
|
|
assert_se(optlen == 16);
|
|
|
|
assert_se(dhcp6_lease_set_sntp(lease, optval,
|
|
|
|
optlen) >= 0);
|
|
|
|
break;
|
|
|
|
|
2014-06-19 14:39:46 +02:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-01-04 14:11:41 +01:00
|
|
|
pos += sizeof(*option) + optlen;
|
|
|
|
}
|
2014-06-19 14:39:46 +02:00
|
|
|
|
2018-01-04 14:11:41 +01:00
|
|
|
assert_se(pos == len);
|
2014-06-19 14:39:46 +02:00
|
|
|
assert_se(opt_clientid);
|
|
|
|
|
2015-01-20 18:35:56 +01:00
|
|
|
sd_dhcp6_lease_reset_address_iter(lease);
|
|
|
|
assert_se(sd_dhcp6_lease_get_address(lease, &addr, <_pref,
|
|
|
|
<_valid) >= 0);
|
2014-06-19 14:39:46 +02:00
|
|
|
assert_se(!memcmp(&addr, &msg_advertise[42], sizeof(addr)));
|
|
|
|
assert_se(lt_pref == 150);
|
|
|
|
assert_se(lt_valid == 180);
|
2015-01-20 18:35:56 +01:00
|
|
|
assert_se(sd_dhcp6_lease_get_address(lease, &addr, <_pref,
|
|
|
|
<_valid) == -ENOMSG);
|
2014-06-19 14:39:46 +02:00
|
|
|
|
2015-01-20 18:35:56 +01:00
|
|
|
sd_dhcp6_lease_reset_address_iter(lease);
|
|
|
|
assert_se(sd_dhcp6_lease_get_address(lease, &addr, <_pref,
|
|
|
|
<_valid) >= 0);
|
2014-06-19 14:39:46 +02:00
|
|
|
assert_se(!memcmp(&addr, &msg_advertise[42], sizeof(addr)));
|
2015-01-20 18:35:56 +01:00
|
|
|
assert_se(sd_dhcp6_lease_get_address(lease, &addr, <_pref,
|
|
|
|
<_valid) == -ENOMSG);
|
|
|
|
sd_dhcp6_lease_reset_address_iter(lease);
|
|
|
|
assert_se(sd_dhcp6_lease_get_address(lease, &addr, <_pref,
|
|
|
|
<_valid) >= 0);
|
2014-06-19 14:39:46 +02:00
|
|
|
assert_se(!memcmp(&addr, &msg_advertise[42], sizeof(addr)));
|
2015-01-20 18:35:56 +01:00
|
|
|
assert_se(sd_dhcp6_lease_get_address(lease, &addr, <_pref,
|
|
|
|
<_valid) == -ENOMSG);
|
2014-06-19 14:39:46 +02:00
|
|
|
|
|
|
|
assert_se(dhcp6_lease_get_serverid(lease, &opt, &len) >= 0);
|
|
|
|
assert_se(len == 14);
|
|
|
|
assert_se(!memcmp(opt, &msg_advertise[179], len));
|
|
|
|
|
|
|
|
assert_se(dhcp6_lease_get_preference(lease, &preference) >= 0);
|
|
|
|
assert_se(preference == 0);
|
|
|
|
|
2015-07-10 10:42:11 +02:00
|
|
|
r = sd_dhcp6_lease_get_dns(lease, &addrs);
|
|
|
|
assert_se(r == 1);
|
|
|
|
assert_se(!memcmp(addrs, &msg_advertise[124], r * 16));
|
|
|
|
|
|
|
|
r = sd_dhcp6_lease_get_domains(lease, &domains);
|
|
|
|
assert_se(r == 1);
|
|
|
|
assert_se(!strcmp("lab.intra", domains[0]));
|
|
|
|
assert_se(domains[1] == NULL);
|
|
|
|
|
|
|
|
r = sd_dhcp6_lease_get_ntp_addrs(lease, &addrs);
|
|
|
|
assert_se(r == 1);
|
|
|
|
assert_se(!memcmp(addrs, &msg_advertise[159], r * 16));
|
|
|
|
|
2014-06-19 14:39:46 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-06-19 14:39:30 +02:00
|
|
|
static int test_hangcheck(sd_event_source *s, uint64_t usec, void *userdata) {
|
|
|
|
assert_not_reached("Test case should have completed in 2 seconds");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-12-10 15:17:33 +01:00
|
|
|
static void test_client_solicit_cb(sd_dhcp6_client *client, int event,
|
|
|
|
void *userdata) {
|
|
|
|
sd_event *e = userdata;
|
2015-07-10 10:42:11 +02:00
|
|
|
sd_dhcp6_lease *lease;
|
|
|
|
struct in6_addr *addrs;
|
|
|
|
char **domains;
|
2014-06-19 14:39:30 +02:00
|
|
|
|
2018-10-18 21:59:18 +02:00
|
|
|
log_debug("/* %s */", __func__);
|
|
|
|
|
2014-12-10 15:17:33 +01:00
|
|
|
assert_se(e);
|
2015-09-22 14:52:23 +02:00
|
|
|
assert_se(event == SD_DHCP6_CLIENT_EVENT_IP_ACQUIRE);
|
2014-06-19 14:39:30 +02:00
|
|
|
|
2015-07-10 10:42:11 +02:00
|
|
|
assert_se(sd_dhcp6_client_get_lease(client, &lease) >= 0);
|
|
|
|
|
|
|
|
assert_se(sd_dhcp6_lease_get_domains(lease, &domains) == 1);
|
|
|
|
assert_se(!strcmp("lab.intra", domains[0]));
|
|
|
|
assert_se(domains[1] == NULL);
|
|
|
|
|
|
|
|
assert_se(sd_dhcp6_lease_get_dns(lease, &addrs) == 1);
|
|
|
|
assert_se(!memcmp(addrs, &msg_advertise[124], 16));
|
|
|
|
|
|
|
|
assert_se(sd_dhcp6_lease_get_ntp_addrs(lease, &addrs) == 1);
|
|
|
|
assert_se(!memcmp(addrs, &msg_advertise[159], 16));
|
|
|
|
|
2016-01-20 14:44:28 +01:00
|
|
|
assert_se(sd_dhcp6_client_set_request_option(client, SD_DHCP6_OPTION_DNS_SERVERS) == -EBUSY);
|
2014-12-10 15:17:33 +01:00
|
|
|
|
|
|
|
sd_event_exit(e, 0);
|
2014-06-19 14:39:30 +02:00
|
|
|
}
|
|
|
|
|
2014-06-19 14:39:54 +02:00
|
|
|
static int test_client_send_reply(DHCP6Message *request) {
|
2014-06-19 14:39:58 +02:00
|
|
|
DHCP6Message reply;
|
|
|
|
|
|
|
|
reply.transaction_id = request->transaction_id;
|
|
|
|
reply.type = DHCP6_REPLY;
|
|
|
|
|
|
|
|
memcpy(msg_reply, &reply.transaction_id, 4);
|
|
|
|
|
|
|
|
memcpy(&msg_reply[26], test_duid, sizeof(test_duid));
|
|
|
|
|
|
|
|
memcpy(&msg_reply[44], &test_iaid, sizeof(test_iaid));
|
|
|
|
|
|
|
|
assert_se(write(test_dhcp_fd[1], msg_reply, sizeof(msg_reply))
|
|
|
|
== sizeof(msg_reply));
|
|
|
|
|
2014-06-19 14:39:54 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-01-04 14:11:41 +01:00
|
|
|
static int test_client_verify_request(DHCP6Message *request, size_t len) {
|
tree-wide: expose "p"-suffix unref calls in public APIs to make gcc cleanup easy
GLIB has recently started to officially support the gcc cleanup
attribute in its public API, hence let's do the same for our APIs.
With this patch we'll define an xyz_unrefp() call for each public
xyz_unref() call, to make it easy to use inside a
__attribute__((cleanup())) expression. Then, all code is ported over to
make use of this.
The new calls are also documented in the man pages, with examples how to
use them (well, I only added docs where the _unref() call itself already
had docs, and the examples, only cover sd_bus_unrefp() and
sd_event_unrefp()).
This also renames sd_lldp_free() to sd_lldp_unref(), since that's how we
tend to call our destructors these days.
Note that this defines no public macro that wraps gcc's attribute and
makes it easier to use. While I think it's our duty in the library to
make our stuff easy to use, I figure it's not our duty to make gcc's own
features easy to use on its own. Most likely, client code which wants to
make use of this should define its own:
#define _cleanup_(function) __attribute__((cleanup(function)))
Or similar, to make the gcc feature easier to use.
Making this logic public has the benefit that we can remove three header
files whose only purpose was to define these functions internally.
See #2008.
2015-11-27 19:13:45 +01:00
|
|
|
_cleanup_(sd_dhcp6_lease_unrefp) sd_dhcp6_lease *lease = NULL;
|
2018-01-04 14:11:41 +01:00
|
|
|
size_t pos = 0;
|
2014-09-01 12:21:34 +02:00
|
|
|
bool found_clientid = false, found_iana = false, found_serverid = false,
|
2017-11-16 10:07:07 +01:00
|
|
|
found_elapsed_time = false, found_fqdn = false;
|
2014-06-19 14:39:54 +02:00
|
|
|
struct in6_addr addr;
|
|
|
|
be32_t val;
|
|
|
|
uint32_t lt_pref, lt_valid;
|
|
|
|
|
2018-10-18 21:59:18 +02:00
|
|
|
log_debug("/* %s */", __func__);
|
2014-06-19 14:39:54 +02:00
|
|
|
|
2018-10-18 21:59:18 +02:00
|
|
|
assert_se(request->type == DHCP6_REQUEST);
|
2014-06-19 14:39:54 +02:00
|
|
|
assert_se(dhcp6_lease_new(&lease) >= 0);
|
|
|
|
|
2018-01-04 14:11:41 +01:00
|
|
|
len -= sizeof(DHCP6Message);
|
|
|
|
|
|
|
|
while (pos < len) {
|
|
|
|
DHCP6Option *option = (DHCP6Option *)&request->options[pos];
|
|
|
|
uint16_t optcode = be16toh(option->code);
|
|
|
|
uint16_t optlen = be16toh(option->len);
|
|
|
|
uint8_t *optval = option->data;
|
|
|
|
|
2014-06-19 14:39:54 +02:00
|
|
|
switch(optcode) {
|
2016-01-20 14:44:28 +01:00
|
|
|
case SD_DHCP6_OPTION_CLIENTID:
|
2014-06-19 14:39:54 +02:00
|
|
|
assert_se(!found_clientid);
|
|
|
|
found_clientid = true;
|
|
|
|
|
|
|
|
assert_se(!memcmp(optval, &test_duid,
|
|
|
|
sizeof(test_duid)));
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
2016-01-20 14:44:28 +01:00
|
|
|
case SD_DHCP6_OPTION_IA_NA:
|
2014-06-19 14:39:54 +02:00
|
|
|
assert_se(!found_iana);
|
|
|
|
found_iana = true;
|
|
|
|
|
|
|
|
assert_se(optlen == 40);
|
|
|
|
assert_se(!memcmp(optval, &test_iaid, sizeof(test_iaid)));
|
|
|
|
|
|
|
|
val = htobe32(80);
|
|
|
|
assert_se(!memcmp(optval + 4, &val, sizeof(val)));
|
|
|
|
|
|
|
|
val = htobe32(120);
|
|
|
|
assert_se(!memcmp(optval + 8, &val, sizeof(val)));
|
|
|
|
|
2018-01-04 14:11:41 +01:00
|
|
|
assert_se(!dhcp6_option_parse_ia(option, &lease->ia));
|
2014-06-19 14:39:54 +02:00
|
|
|
|
|
|
|
break;
|
|
|
|
|
2016-01-20 14:44:28 +01:00
|
|
|
case SD_DHCP6_OPTION_SERVERID:
|
2014-06-19 14:39:54 +02:00
|
|
|
assert_se(!found_serverid);
|
|
|
|
found_serverid = true;
|
|
|
|
|
|
|
|
assert_se(optlen == 14);
|
|
|
|
assert_se(!memcmp(&msg_advertise[179], optval, optlen));
|
|
|
|
|
2014-09-01 12:21:34 +02:00
|
|
|
break;
|
|
|
|
|
2016-01-20 14:44:28 +01:00
|
|
|
case SD_DHCP6_OPTION_ELAPSED_TIME:
|
2014-09-01 12:21:34 +02:00
|
|
|
assert_se(!found_elapsed_time);
|
|
|
|
found_elapsed_time = true;
|
|
|
|
|
|
|
|
assert_se(optlen == 2);
|
|
|
|
|
2017-11-16 10:07:07 +01:00
|
|
|
break;
|
|
|
|
case SD_DHCP6_OPTION_FQDN:
|
|
|
|
assert_se(!found_fqdn);
|
|
|
|
found_fqdn = true;
|
|
|
|
|
|
|
|
assert_se(optlen == 17);
|
|
|
|
|
|
|
|
assert_se(optval[0] == 0x01);
|
|
|
|
assert_se(!memcmp(optval + 1, fqdn_wire, sizeof(fqdn_wire)));
|
2014-06-19 14:39:54 +02:00
|
|
|
break;
|
|
|
|
}
|
2018-01-04 14:11:41 +01:00
|
|
|
|
|
|
|
pos += sizeof(*option) + optlen;
|
2014-06-19 14:39:54 +02:00
|
|
|
}
|
|
|
|
|
2014-09-01 12:21:34 +02:00
|
|
|
assert_se(found_clientid && found_iana && found_serverid &&
|
|
|
|
found_elapsed_time);
|
2014-06-19 14:39:54 +02:00
|
|
|
|
2015-01-20 18:35:56 +01:00
|
|
|
sd_dhcp6_lease_reset_address_iter(lease);
|
|
|
|
assert_se(sd_dhcp6_lease_get_address(lease, &addr, <_pref,
|
|
|
|
<_valid) >= 0);
|
2014-06-19 14:39:54 +02:00
|
|
|
assert_se(!memcmp(&addr, &msg_advertise[42], sizeof(addr)));
|
|
|
|
assert_se(lt_pref == 150);
|
|
|
|
assert_se(lt_valid == 180);
|
|
|
|
|
2015-01-20 18:35:56 +01:00
|
|
|
assert_se(sd_dhcp6_lease_get_address(lease, &addr, <_pref,
|
|
|
|
<_valid) == -ENOMSG);
|
2014-06-19 14:39:54 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-08-27 17:48:24 +02:00
|
|
|
static int test_client_send_advertise(DHCP6Message *solicit) {
|
2014-06-19 14:39:54 +02:00
|
|
|
DHCP6Message advertise;
|
|
|
|
|
|
|
|
advertise.transaction_id = solicit->transaction_id;
|
|
|
|
advertise.type = DHCP6_ADVERTISE;
|
|
|
|
|
|
|
|
memcpy(msg_advertise, &advertise.transaction_id, 4);
|
|
|
|
|
|
|
|
memcpy(&msg_advertise[8], test_duid, sizeof(test_duid));
|
|
|
|
|
|
|
|
memcpy(&msg_advertise[26], &test_iaid, sizeof(test_iaid));
|
|
|
|
|
|
|
|
assert_se(write(test_dhcp_fd[1], msg_advertise, sizeof(msg_advertise))
|
|
|
|
== sizeof(msg_advertise));
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-01-04 14:11:41 +01:00
|
|
|
static int test_client_verify_solicit(DHCP6Message *solicit, size_t len) {
|
2014-09-01 12:21:34 +02:00
|
|
|
bool found_clientid = false, found_iana = false,
|
2017-11-16 10:07:07 +01:00
|
|
|
found_elapsed_time = false, found_fqdn = false;
|
2018-01-04 14:11:41 +01:00
|
|
|
size_t pos = 0;
|
2014-06-19 14:39:30 +02:00
|
|
|
|
2018-10-18 21:59:18 +02:00
|
|
|
log_debug("/* %s */", __func__);
|
|
|
|
|
2014-06-19 14:39:30 +02:00
|
|
|
assert_se(solicit->type == DHCP6_SOLICIT);
|
|
|
|
|
2018-01-04 14:11:41 +01:00
|
|
|
len -= sizeof(DHCP6Message);
|
|
|
|
|
|
|
|
while (pos < len) {
|
|
|
|
DHCP6Option *option = (DHCP6Option *)&solicit->options[pos];
|
|
|
|
uint16_t optcode = be16toh(option->code);
|
|
|
|
uint16_t optlen = be16toh(option->len);
|
|
|
|
uint8_t *optval = option->data;
|
|
|
|
|
2014-06-19 14:39:30 +02:00
|
|
|
switch(optcode) {
|
2016-01-20 14:44:28 +01:00
|
|
|
case SD_DHCP6_OPTION_CLIENTID:
|
2014-06-19 14:39:30 +02:00
|
|
|
assert_se(!found_clientid);
|
|
|
|
found_clientid = true;
|
|
|
|
|
2014-06-19 14:39:54 +02:00
|
|
|
assert_se(optlen == sizeof(test_duid));
|
|
|
|
memcpy(&test_duid, optval, sizeof(test_duid));
|
2014-06-19 14:39:30 +02:00
|
|
|
|
|
|
|
break;
|
|
|
|
|
2016-01-20 14:44:28 +01:00
|
|
|
case SD_DHCP6_OPTION_IA_NA:
|
2014-06-19 14:39:30 +02:00
|
|
|
assert_se(!found_iana);
|
|
|
|
found_iana = true;
|
|
|
|
|
|
|
|
assert_se(optlen == 12);
|
|
|
|
|
2014-06-19 14:39:54 +02:00
|
|
|
memcpy(&test_iaid, optval, sizeof(test_iaid));
|
|
|
|
|
2014-09-01 12:21:34 +02:00
|
|
|
break;
|
|
|
|
|
2016-01-20 14:44:28 +01:00
|
|
|
case SD_DHCP6_OPTION_ELAPSED_TIME:
|
2014-09-01 12:21:34 +02:00
|
|
|
assert_se(!found_elapsed_time);
|
|
|
|
found_elapsed_time = true;
|
|
|
|
|
|
|
|
assert_se(optlen == 2);
|
|
|
|
|
2017-11-16 10:07:07 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SD_DHCP6_OPTION_FQDN:
|
|
|
|
assert_se(!found_fqdn);
|
|
|
|
found_fqdn = true;
|
|
|
|
|
|
|
|
assert_se(optlen == 17);
|
|
|
|
|
|
|
|
assert_se(optval[0] == 0x01);
|
|
|
|
assert_se(!memcmp(optval + 1, fqdn_wire, sizeof(fqdn_wire)));
|
|
|
|
|
2014-06-19 14:39:30 +02:00
|
|
|
break;
|
|
|
|
}
|
2018-01-04 14:11:41 +01:00
|
|
|
|
|
|
|
pos += sizeof(*option) + optlen;
|
2014-06-19 14:39:30 +02:00
|
|
|
}
|
|
|
|
|
2018-01-04 14:11:41 +01:00
|
|
|
assert_se(pos == len);
|
2014-09-01 12:21:34 +02:00
|
|
|
assert_se(found_clientid && found_iana && found_elapsed_time);
|
2014-06-19 14:39:30 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-12-10 15:17:33 +01:00
|
|
|
static void test_client_information_cb(sd_dhcp6_client *client, int event,
|
|
|
|
void *userdata) {
|
|
|
|
sd_event *e = userdata;
|
2015-07-10 10:42:11 +02:00
|
|
|
sd_dhcp6_lease *lease;
|
|
|
|
struct in6_addr *addrs;
|
2015-11-16 17:43:08 +01:00
|
|
|
struct in6_addr address = { { { 0xfe, 0x80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0x01 } } };
|
2015-07-10 10:42:11 +02:00
|
|
|
char **domains;
|
2014-12-10 15:17:33 +01:00
|
|
|
|
2018-10-18 21:59:18 +02:00
|
|
|
log_debug("/* %s */", __func__);
|
|
|
|
|
2014-12-10 15:17:33 +01:00
|
|
|
assert_se(e);
|
2015-09-22 14:52:23 +02:00
|
|
|
assert_se(event == SD_DHCP6_CLIENT_EVENT_INFORMATION_REQUEST);
|
2014-12-10 15:17:33 +01:00
|
|
|
|
2015-07-10 10:42:11 +02:00
|
|
|
assert_se(sd_dhcp6_client_get_lease(client, &lease) >= 0);
|
|
|
|
|
|
|
|
assert_se(sd_dhcp6_lease_get_domains(lease, &domains) == 1);
|
|
|
|
assert_se(!strcmp("lab.intra", domains[0]));
|
|
|
|
assert_se(domains[1] == NULL);
|
|
|
|
|
|
|
|
assert_se(sd_dhcp6_lease_get_dns(lease, &addrs) == 1);
|
|
|
|
assert_se(!memcmp(addrs, &msg_advertise[124], 16));
|
|
|
|
|
|
|
|
assert_se(sd_dhcp6_lease_get_ntp_addrs(lease, &addrs) == 1);
|
|
|
|
assert_se(!memcmp(addrs, &msg_advertise[159], 16));
|
|
|
|
|
2015-09-23 13:00:03 +02:00
|
|
|
assert_se(sd_dhcp6_client_set_information_request(client, false) == -EBUSY);
|
|
|
|
assert_se(sd_dhcp6_client_set_callback(client, NULL, e) >= 0);
|
|
|
|
assert_se(sd_dhcp6_client_stop(client) >= 0);
|
2014-12-10 15:17:33 +01:00
|
|
|
assert_se(sd_dhcp6_client_set_information_request(client, false) >= 0);
|
2015-09-23 13:00:03 +02:00
|
|
|
|
2014-12-10 15:17:33 +01:00
|
|
|
assert_se(sd_dhcp6_client_set_callback(client,
|
|
|
|
test_client_solicit_cb, e) >= 0);
|
|
|
|
|
2015-11-16 17:43:08 +01:00
|
|
|
assert_se(sd_dhcp6_client_set_local_address(client, &address) >= 0);
|
|
|
|
|
2014-12-10 15:17:33 +01:00
|
|
|
assert_se(sd_dhcp6_client_start(client) >= 0);
|
2018-01-04 14:11:41 +01:00
|
|
|
|
2014-12-10 15:17:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static int test_client_verify_information_request(DHCP6Message *information_request,
|
2018-01-04 14:11:41 +01:00
|
|
|
size_t len) {
|
2014-12-10 15:17:33 +01:00
|
|
|
|
tree-wide: expose "p"-suffix unref calls in public APIs to make gcc cleanup easy
GLIB has recently started to officially support the gcc cleanup
attribute in its public API, hence let's do the same for our APIs.
With this patch we'll define an xyz_unrefp() call for each public
xyz_unref() call, to make it easy to use inside a
__attribute__((cleanup())) expression. Then, all code is ported over to
make use of this.
The new calls are also documented in the man pages, with examples how to
use them (well, I only added docs where the _unref() call itself already
had docs, and the examples, only cover sd_bus_unrefp() and
sd_event_unrefp()).
This also renames sd_lldp_free() to sd_lldp_unref(), since that's how we
tend to call our destructors these days.
Note that this defines no public macro that wraps gcc's attribute and
makes it easier to use. While I think it's our duty in the library to
make our stuff easy to use, I figure it's not our duty to make gcc's own
features easy to use on its own. Most likely, client code which wants to
make use of this should define its own:
#define _cleanup_(function) __attribute__((cleanup(function)))
Or similar, to make the gcc feature easier to use.
Making this logic public has the benefit that we can remove three header
files whose only purpose was to define these functions internally.
See #2008.
2015-11-27 19:13:45 +01:00
|
|
|
_cleanup_(sd_dhcp6_lease_unrefp) sd_dhcp6_lease *lease = NULL;
|
2018-01-04 14:11:41 +01:00
|
|
|
size_t pos = 0;
|
2014-12-10 15:17:33 +01:00
|
|
|
bool found_clientid = false, found_elapsed_time = false;
|
|
|
|
struct in6_addr addr;
|
|
|
|
uint32_t lt_pref, lt_valid;
|
|
|
|
|
2018-10-18 21:59:18 +02:00
|
|
|
log_debug("/* %s */", __func__);
|
2014-12-10 15:17:33 +01:00
|
|
|
|
2018-10-18 21:59:18 +02:00
|
|
|
assert_se(information_request->type == DHCP6_INFORMATION_REQUEST);
|
2014-12-10 15:17:33 +01:00
|
|
|
assert_se(dhcp6_lease_new(&lease) >= 0);
|
|
|
|
|
2018-01-04 14:11:41 +01:00
|
|
|
len -= sizeof(DHCP6Message);
|
|
|
|
|
|
|
|
while (pos < len) {
|
|
|
|
DHCP6Option *option = (DHCP6Option *)&information_request->options[pos];
|
|
|
|
uint16_t optcode = be16toh(option->code);
|
|
|
|
uint16_t optlen = be16toh(option->len);
|
|
|
|
uint8_t *optval = option->data;
|
|
|
|
|
2014-12-10 15:17:33 +01:00
|
|
|
switch(optcode) {
|
2016-01-20 14:44:28 +01:00
|
|
|
case SD_DHCP6_OPTION_CLIENTID:
|
2014-12-10 15:17:33 +01:00
|
|
|
assert_se(!found_clientid);
|
|
|
|
found_clientid = true;
|
|
|
|
|
|
|
|
assert_se(optlen == sizeof(test_duid));
|
|
|
|
memcpy(&test_duid, optval, sizeof(test_duid));
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
2016-01-20 14:44:28 +01:00
|
|
|
case SD_DHCP6_OPTION_IA_NA:
|
2014-12-10 15:17:33 +01:00
|
|
|
assert_not_reached("IA TA option must not be present");
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
2016-01-20 14:44:28 +01:00
|
|
|
case SD_DHCP6_OPTION_SERVERID:
|
2014-12-10 15:17:33 +01:00
|
|
|
assert_not_reached("Server ID option must not be present");
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
2016-01-20 14:44:28 +01:00
|
|
|
case SD_DHCP6_OPTION_ELAPSED_TIME:
|
2014-12-10 15:17:33 +01:00
|
|
|
assert_se(!found_elapsed_time);
|
|
|
|
found_elapsed_time = true;
|
|
|
|
|
|
|
|
assert_se(optlen == 2);
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
2018-01-04 14:11:41 +01:00
|
|
|
|
|
|
|
pos += sizeof(*option) + optlen;
|
2014-12-10 15:17:33 +01:00
|
|
|
}
|
|
|
|
|
2018-01-04 14:11:41 +01:00
|
|
|
assert_se(pos == len);
|
2014-12-10 15:17:33 +01:00
|
|
|
assert_se(found_clientid && found_elapsed_time);
|
|
|
|
|
2015-01-20 18:35:56 +01:00
|
|
|
sd_dhcp6_lease_reset_address_iter(lease);
|
2014-12-10 15:17:33 +01:00
|
|
|
|
2015-01-20 18:35:56 +01:00
|
|
|
assert_se(sd_dhcp6_lease_get_address(lease, &addr, <_pref,
|
|
|
|
<_valid) == -ENOMSG);
|
2014-12-10 15:17:33 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-06-19 14:39:30 +02:00
|
|
|
int dhcp6_network_send_udp_socket(int s, struct in6_addr *server_address,
|
|
|
|
const void *packet, size_t len) {
|
|
|
|
struct in6_addr mcast =
|
|
|
|
IN6ADDR_ALL_DHCP6_RELAY_AGENTS_AND_SERVERS_INIT;
|
|
|
|
DHCP6Message *message;
|
|
|
|
|
|
|
|
assert_se(s == test_dhcp_fd[0]);
|
|
|
|
assert_se(server_address);
|
|
|
|
assert_se(packet);
|
|
|
|
assert_se(len > sizeof(DHCP6Message) + 4);
|
|
|
|
assert_se(IN6_ARE_ADDR_EQUAL(server_address, &mcast));
|
|
|
|
|
|
|
|
message = (DHCP6Message *)packet;
|
|
|
|
|
|
|
|
assert_se(message->transaction_id & 0x00ffffff);
|
|
|
|
|
2014-06-19 14:39:54 +02:00
|
|
|
if (test_client_message_num == 0) {
|
2018-01-04 14:11:41 +01:00
|
|
|
test_client_verify_information_request(message, len);
|
2014-12-10 15:17:33 +01:00
|
|
|
test_client_send_reply(message);
|
|
|
|
test_client_message_num++;
|
|
|
|
} else if (test_client_message_num == 1) {
|
2018-01-04 14:11:41 +01:00
|
|
|
test_client_verify_solicit(message, len);
|
2014-06-19 14:39:54 +02:00
|
|
|
test_client_send_advertise(message);
|
|
|
|
test_client_message_num++;
|
2014-12-10 15:17:33 +01:00
|
|
|
} else if (test_client_message_num == 2) {
|
2018-01-04 14:11:41 +01:00
|
|
|
test_client_verify_request(message, len);
|
2014-06-19 14:39:54 +02:00
|
|
|
test_client_send_reply(message);
|
|
|
|
test_client_message_num++;
|
|
|
|
}
|
2014-06-19 14:39:30 +02:00
|
|
|
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
2014-12-10 15:17:33 +01:00
|
|
|
int dhcp6_network_bind_udp_socket(int index, struct in6_addr *local_address) {
|
|
|
|
assert_se(index == test_index);
|
2014-06-24 15:20:32 +02:00
|
|
|
|
2018-11-11 18:03:22 +01:00
|
|
|
if (socketpair(AF_UNIX, SOCK_STREAM | SOCK_CLOEXEC | SOCK_NONBLOCK, 0, test_dhcp_fd) < 0)
|
2014-12-10 15:17:33 +01:00
|
|
|
return -errno;
|
2014-06-19 14:39:30 +02:00
|
|
|
|
2014-12-10 15:17:33 +01:00
|
|
|
return test_dhcp_fd[0];
|
2014-06-19 14:39:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static int test_client_solicit(sd_event *e) {
|
|
|
|
sd_dhcp6_client *client;
|
2014-07-24 18:53:01 +02:00
|
|
|
usec_t time_now = now(clock_boottime_or_monotonic());
|
2015-11-16 17:43:08 +01:00
|
|
|
struct in6_addr address = { { { 0xfe, 0x80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0x01 } } };
|
2018-09-07 17:24:15 +02:00
|
|
|
int val;
|
2014-06-19 14:39:30 +02:00
|
|
|
|
2018-10-18 21:59:18 +02:00
|
|
|
log_debug("/* %s */", __func__);
|
2014-06-19 14:39:30 +02:00
|
|
|
|
|
|
|
assert_se(sd_dhcp6_client_new(&client) >= 0);
|
|
|
|
assert_se(client);
|
|
|
|
|
|
|
|
assert_se(sd_dhcp6_client_attach_event(client, e, 0) >= 0);
|
|
|
|
|
2016-05-23 16:13:18 +02:00
|
|
|
assert_se(sd_dhcp6_client_set_ifindex(client, test_index) == 0);
|
2014-10-08 21:15:45 +02:00
|
|
|
assert_se(sd_dhcp6_client_set_mac(client, (const uint8_t *) &mac_addr,
|
|
|
|
sizeof (mac_addr),
|
|
|
|
ARPHRD_ETHER) >= 0);
|
2017-11-16 10:07:07 +01:00
|
|
|
assert_se(sd_dhcp6_client_set_fqdn(client, "host.lab.intra") == 1);
|
2014-06-19 14:39:30 +02:00
|
|
|
|
2014-12-10 15:17:33 +01:00
|
|
|
assert_se(sd_dhcp6_client_get_information_request(client, &val) >= 0);
|
2018-09-07 17:24:15 +02:00
|
|
|
assert_se(val == 0);
|
|
|
|
assert_se(sd_dhcp6_client_set_information_request(client, 42) >= 0);
|
2014-12-10 15:17:33 +01:00
|
|
|
assert_se(sd_dhcp6_client_get_information_request(client, &val) >= 0);
|
2018-09-07 17:24:15 +02:00
|
|
|
assert_se(val);
|
2014-12-10 15:17:33 +01:00
|
|
|
|
2014-06-19 14:39:30 +02:00
|
|
|
assert_se(sd_dhcp6_client_set_callback(client,
|
2014-12-10 15:17:33 +01:00
|
|
|
test_client_information_cb, e) >= 0);
|
2014-06-19 14:39:30 +02:00
|
|
|
|
2014-07-24 18:53:01 +02:00
|
|
|
assert_se(sd_event_add_time(e, &hangcheck, clock_boottime_or_monotonic(),
|
2014-06-19 14:39:30 +02:00
|
|
|
time_now + 2 * USEC_PER_SEC, 0,
|
|
|
|
test_hangcheck, NULL) >= 0);
|
|
|
|
|
2015-11-16 17:43:08 +01:00
|
|
|
assert_se(sd_dhcp6_client_set_local_address(client, &address) >= 0);
|
|
|
|
|
2014-06-19 14:39:30 +02:00
|
|
|
assert_se(sd_dhcp6_client_start(client) >= 0);
|
|
|
|
|
|
|
|
sd_event_loop(e);
|
|
|
|
|
|
|
|
hangcheck = sd_event_source_unref(hangcheck);
|
|
|
|
|
|
|
|
assert_se(!sd_dhcp6_client_unref(client));
|
|
|
|
|
|
|
|
test_dhcp_fd[1] = safe_close(test_dhcp_fd[1]);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-06-19 14:39:12 +02:00
|
|
|
int main(int argc, char *argv[]) {
|
tree-wide: expose "p"-suffix unref calls in public APIs to make gcc cleanup easy
GLIB has recently started to officially support the gcc cleanup
attribute in its public API, hence let's do the same for our APIs.
With this patch we'll define an xyz_unrefp() call for each public
xyz_unref() call, to make it easy to use inside a
__attribute__((cleanup())) expression. Then, all code is ported over to
make use of this.
The new calls are also documented in the man pages, with examples how to
use them (well, I only added docs where the _unref() call itself already
had docs, and the examples, only cover sd_bus_unrefp() and
sd_event_unrefp()).
This also renames sd_lldp_free() to sd_lldp_unref(), since that's how we
tend to call our destructors these days.
Note that this defines no public macro that wraps gcc's attribute and
makes it easier to use. While I think it's our duty in the library to
make our stuff easy to use, I figure it's not our duty to make gcc's own
features easy to use on its own. Most likely, client code which wants to
make use of this should define its own:
#define _cleanup_(function) __attribute__((cleanup(function)))
Or similar, to make the gcc feature easier to use.
Making this logic public has the benefit that we can remove three header
files whose only purpose was to define these functions internally.
See #2008.
2015-11-27 19:13:45 +01:00
|
|
|
_cleanup_(sd_event_unrefp) sd_event *e;
|
2014-06-19 14:39:12 +02:00
|
|
|
|
|
|
|
assert_se(sd_event_new(&e) >= 0);
|
|
|
|
|
2018-09-13 14:31:13 +02:00
|
|
|
test_setup_logging(LOG_DEBUG);
|
2014-06-19 14:39:12 +02:00
|
|
|
|
|
|
|
test_client_basic(e);
|
2014-06-19 14:39:20 +02:00
|
|
|
test_option(e);
|
2018-01-04 14:11:44 +01:00
|
|
|
test_option_status(e);
|
2014-06-19 14:39:46 +02:00
|
|
|
test_advertise_option(e);
|
2014-06-19 14:39:30 +02:00
|
|
|
test_client_solicit(e);
|
2014-06-19 14:39:20 +02:00
|
|
|
|
2014-06-19 14:39:12 +02:00
|
|
|
return 0;
|
|
|
|
}
|