1 /* SPDX-License-Identifier: LGPL-2.1+ */
3 Copyright © 2017 Intel Corporation. All rights reserved.
6 #include <netinet/icmp6.h>
11 #include "alloc-util.h"
12 #include "hexdecoct.h"
13 #include "icmp6-util.h"
14 #include "socket-util.h"
18 static struct ether_addr mac_addr
= {
19 .ether_addr_octet
= { 0x78, 0x2b, 0xcb, 0xb3, 0x6d, 0x53 }
22 static uint8_t advertisement
[] = {
23 /* ICMPv6 Router Advertisement, no checksum */
24 0x86, 0x00, 0x00, 0x00, 0x40, 0xc0, 0x00, 0xb4,
25 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
26 /* Source Link Layer Address Option */
27 0x01, 0x01, 0x78, 0x2b, 0xcb, 0xb3, 0x6d, 0x53,
28 /* Prefix Information Option */
29 0x03, 0x04, 0x40, 0xc0, 0x00, 0x00, 0x01, 0xf4,
30 0x00, 0x00, 0x01, 0xb8, 0x00, 0x00, 0x00, 0x00,
31 0x20, 0x01, 0x0d, 0xb8, 0xde, 0xad, 0xbe, 0xef,
32 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
33 /* Prefix Information Option */
34 0x03, 0x04, 0x40, 0xc0, 0x00, 0x27, 0x8d, 0x00,
35 0x00, 0x09, 0x3a, 0x80, 0x00, 0x00, 0x00, 0x00,
36 0x20, 0x01, 0x0d, 0xb8, 0x0b, 0x16, 0xd0, 0x0d,
37 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
38 /* Prefix Information Option */
39 0x03, 0x04, 0x30, 0xc0, 0x00, 0x27, 0x8d, 0x00,
40 0x00, 0x09, 0x3a, 0x80, 0x00, 0x00, 0x00, 0x00,
41 0x20, 0x01, 0x0d, 0xb8, 0xc0, 0x01, 0x0d, 0xad,
42 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
43 /* Recursive DNS Server Option */
44 0x19, 0x03, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3c,
45 0x20, 0x01, 0x0d, 0xb8, 0xde, 0xad, 0xbe, 0xef,
46 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01,
47 /* DNS Search List Option */
48 0x1f, 0x03, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3c,
49 0x03, 0x6c, 0x61, 0x62, 0x05, 0x69, 0x6e, 0x74,
50 0x72, 0x61, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
53 static sd_event_source
*test_hangcheck
;
54 static bool test_stopped
;
55 static int test_fd
[2];
56 static sd_event_source
*recv_router_advertisement
;
58 struct in6_addr address
;
59 unsigned char prefixlen
;
64 { { { { 0x20, 0x01, 0x0d, 0xb8, 0xde, 0xad, 0xbe, 0xef,
65 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 } } }, 64,
67 { { { { 0x20, 0x01, 0x0d, 0xb8, 0x0b, 0x16, 0xd0, 0x0d,
68 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 } } }, 64,
69 /* indicate default valid and preferred lifetimes for the test code */
71 { { { { 0x20, 0x01, 0x0d, 0xb8, 0x0b, 0x16, 0xd0, 0x0d,
72 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 } } }, 58,
74 /* indicate that this prefix already exists */
76 { { { { 0x20, 0x01, 0x0d, 0xb8, 0x0b, 0x16, 0xd0, 0x0d,
77 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 } } }, 120,
79 /* indicate that this prefix already exists */
81 { { { { 0x20, 0x01, 0x0d, 0xb8, 0x0b, 0x16, 0xd0, 0x0d,
82 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 } } }, 12,
84 /* indicate that this prefix already exists */
86 { { { { 0x20, 0x01, 0x0d, 0xb8, 0xc0, 0x01, 0x0d, 0xad,
87 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 } } }, 48,
89 { { { { 0x20, 0x01, 0x0d, 0xb8, 0xc0, 0x01, 0x0d, 0xad,
90 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 } } }, 60,
92 /* indicate that this prefix already exists */
96 static const struct in6_addr test_rdnss
= { { { 0x20, 0x01, 0x0d, 0xb8,
97 0xde, 0xad, 0xbe, 0xef,
98 0x00, 0x00, 0x00, 0x00,
99 0x00, 0x00, 0x00, 0x01 } } };
100 static const char *test_dnssl
[] = { "lab.intra",
103 static int test_rs_hangcheck(sd_event_source
*s
, uint64_t usec
,
110 static void test_radv_prefix(void) {
113 printf("* %s\n", __FUNCTION__
);
115 assert_se(sd_radv_prefix_new(&p
) >= 0);
117 assert_se(sd_radv_prefix_set_onlink(NULL
, true) < 0);
118 assert_se(sd_radv_prefix_set_onlink(p
, true) >= 0);
119 assert_se(sd_radv_prefix_set_onlink(p
, false) >= 0);
121 assert_se(sd_radv_prefix_set_address_autoconfiguration(NULL
, true) < 0);
122 assert_se(sd_radv_prefix_set_address_autoconfiguration(p
, true) >= 0);
123 assert_se(sd_radv_prefix_set_address_autoconfiguration(p
, false) >= 0);
125 assert_se(sd_radv_prefix_set_valid_lifetime(NULL
, true) < 0);
126 assert_se(sd_radv_prefix_set_valid_lifetime(p
, ~0) >= 0);
127 assert_se(sd_radv_prefix_set_valid_lifetime(p
, 42) >= 0);
128 assert_se(sd_radv_prefix_set_valid_lifetime(p
, 0) >= 0);
130 assert_se(sd_radv_prefix_set_preferred_lifetime(NULL
, true) < 0);
131 assert_se(sd_radv_prefix_set_preferred_lifetime(p
, ~0) >= 0);
132 assert_se(sd_radv_prefix_set_preferred_lifetime(p
, 42) >= 0);
133 assert_se(sd_radv_prefix_set_preferred_lifetime(p
, 0) >= 0);
135 assert_se(sd_radv_prefix_set_prefix(NULL
, NULL
, 0) < 0);
136 assert_se(sd_radv_prefix_set_prefix(p
, NULL
, 0) < 0);
138 assert_se(sd_radv_prefix_set_prefix(p
, &prefix
[0].address
, 64) >= 0);
139 assert_se(sd_radv_prefix_set_prefix(p
, &prefix
[0].address
, 0) < 0);
140 assert_se(sd_radv_prefix_set_prefix(p
, &prefix
[0].address
, 1) < 0);
141 assert_se(sd_radv_prefix_set_prefix(p
, &prefix
[0].address
, 2) < 0);
142 assert_se(sd_radv_prefix_set_prefix(p
, &prefix
[0].address
, 3) >= 0);
143 assert_se(sd_radv_prefix_set_prefix(p
, &prefix
[0].address
, 125) >= 0);
144 assert_se(sd_radv_prefix_set_prefix(p
, &prefix
[0].address
, 128) >= 0);
145 assert_se(sd_radv_prefix_set_prefix(p
, &prefix
[0].address
, 129) < 0);
146 assert_se(sd_radv_prefix_set_prefix(p
, &prefix
[0].address
, 255) < 0);
148 p
= sd_radv_prefix_unref(p
);
152 static void test_radv(void) {
155 printf("* %s\n", __FUNCTION__
);
157 assert_se(sd_radv_new(&ra
) >= 0);
160 assert_se(sd_radv_set_ifindex(NULL
, 0) < 0);
161 assert_se(sd_radv_set_ifindex(ra
, 0) >= 0);
162 assert_se(sd_radv_set_ifindex(ra
, -1) >= 0);
163 assert_se(sd_radv_set_ifindex(ra
, -2) < 0);
164 assert_se(sd_radv_set_ifindex(ra
, 42) >= 0);
166 assert_se(sd_radv_set_mac(NULL
, NULL
) < 0);
167 assert_se(sd_radv_set_mac(ra
, NULL
) >= 0);
168 assert_se(sd_radv_set_mac(ra
, &mac_addr
) >= 0);
170 assert_se(sd_radv_set_mtu(NULL
, 0) < 0);
171 assert_se(sd_radv_set_mtu(ra
, 0) < 0);
172 assert_se(sd_radv_set_mtu(ra
, 1279) < 0);
173 assert_se(sd_radv_set_mtu(ra
, 1280) >= 0);
174 assert_se(sd_radv_set_mtu(ra
, ~0) >= 0);
176 assert_se(sd_radv_set_hop_limit(NULL
, 0) < 0);
177 assert_se(sd_radv_set_hop_limit(ra
, 0) >= 0);
178 assert_se(sd_radv_set_hop_limit(ra
, ~0) >= 0);
180 assert_se(sd_radv_set_router_lifetime(NULL
, 0) < 0);
181 assert_se(sd_radv_set_router_lifetime(ra
, 0) >= 0);
182 assert_se(sd_radv_set_router_lifetime(ra
, ~0) >= 0);
184 assert_se(sd_radv_set_preference(NULL
, 0) < 0);
185 assert_se(sd_radv_set_preference(ra
, SD_NDISC_PREFERENCE_LOW
) >= 0);
186 assert_se(sd_radv_set_preference(ra
, SD_NDISC_PREFERENCE_MEDIUM
) >= 0);
187 assert_se(sd_radv_set_preference(ra
, SD_NDISC_PREFERENCE_HIGH
) >= 0);
188 assert_se(sd_radv_set_preference(ra
, ~0) < 0);
190 assert_se(sd_radv_set_preference(ra
, SD_NDISC_PREFERENCE_HIGH
) >= 0);
191 assert_se(sd_radv_set_router_lifetime(ra
, 42000) >= 0);
192 assert_se(sd_radv_set_router_lifetime(ra
, 0) < 0);
193 assert_se(sd_radv_set_preference(ra
, SD_NDISC_PREFERENCE_MEDIUM
) >= 0);
194 assert_se(sd_radv_set_router_lifetime(ra
, 0) >= 0);
196 assert_se(sd_radv_set_managed_information(NULL
, true) < 0);
197 assert_se(sd_radv_set_managed_information(ra
, true) >= 0);
198 assert_se(sd_radv_set_managed_information(ra
, false) >= 0);
200 assert_se(sd_radv_set_other_information(NULL
, true) < 0);
201 assert_se(sd_radv_set_other_information(ra
, true) >= 0);
202 assert_se(sd_radv_set_other_information(ra
, false) >= 0);
204 assert_se(sd_radv_set_rdnss(NULL
, 0, NULL
, 0) < 0);
205 assert_se(sd_radv_set_rdnss(ra
, 0, NULL
, 0) >= 0);
206 assert_se(sd_radv_set_rdnss(ra
, 0, NULL
, 128) < 0);
207 assert_se(sd_radv_set_rdnss(ra
, 600, &test_rdnss
, 0) >= 0);
208 assert_se(sd_radv_set_rdnss(ra
, 600, &test_rdnss
, 1) >= 0);
209 assert_se(sd_radv_set_rdnss(ra
, 0, &test_rdnss
, 1) >= 0);
210 assert_se(sd_radv_set_rdnss(ra
, 0, NULL
, 0) >= 0);
212 assert_se(sd_radv_set_dnssl(ra
, 0, NULL
) >= 0);
213 assert_se(sd_radv_set_dnssl(ra
, 600, NULL
) >= 0);
214 assert_se(sd_radv_set_dnssl(ra
, 0, (char **)test_dnssl
) >= 0);
215 assert_se(sd_radv_set_dnssl(ra
, 600, (char **)test_dnssl
) >= 0);
217 ra
= sd_radv_unref(ra
);
221 int icmp6_bind_router_solicitation(int index
) {
225 int icmp6_bind_router_advertisement(int index
) {
226 assert_se(index
== 42);
231 int icmp6_send_router_solicitation(int s
, const struct ether_addr
*ether_addr
) {
236 int icmp6_receive(int fd
, void *iov_base
, size_t iov_len
,
237 struct in6_addr
*dst
, triple_timestamp
*timestamp
) {
238 assert_se(read (fd
, iov_base
, iov_len
) == (ssize_t
)iov_len
);
241 triple_timestamp_get(timestamp
);
246 static int radv_recv(sd_event_source
*s
, int fd
, uint32_t revents
, void *userdata
) {
247 sd_radv
*ra
= userdata
;
248 unsigned char buf
[168];
251 assert_se(read(test_fd
[0], &buf
, sizeof(buf
)) == sizeof(buf
));
253 /* router lifetime must be zero when test is stopped */
255 advertisement
[6] = 0x00;
256 advertisement
[7] = 0x00;
259 printf ("Received Router Advertisement with lifetime %u\n",
260 (advertisement
[6] << 8) + advertisement
[7]);
262 /* test only up to buf size, rest is not yet implemented */
263 for (i
= 0; i
< sizeof(buf
); i
++) {
267 printf("0x%02x", buf
[i
]);
269 assert_se(buf
[i
] == advertisement
[i
]);
280 e
= sd_radv_get_event(ra
);
286 assert_se(sd_radv_stop(ra
) >= 0);
292 static void test_ra(void) {
295 usec_t time_now
= now(clock_boottime_or_monotonic());
298 printf("* %s\n", __FUNCTION__
);
300 assert_se(socketpair(AF_UNIX
, SOCK_SEQPACKET
, 0, test_fd
) >= 0);
302 assert_se(sd_event_new(&e
) >= 0);
304 assert_se(sd_radv_new(&ra
) >= 0);
307 assert_se(sd_radv_attach_event(ra
, e
, 0) >= 0);
309 assert_se(sd_radv_set_ifindex(ra
, 42) >= 0);
310 assert_se(sd_radv_set_mac(ra
, &mac_addr
) >= 0);
311 assert_se(sd_radv_set_router_lifetime(ra
, 180) >= 0);
312 assert_se(sd_radv_set_hop_limit(ra
, 64) >= 0);
313 assert_se(sd_radv_set_managed_information(ra
, true) >= 0);
314 assert_se(sd_radv_set_other_information(ra
, true) >= 0);
315 assert_se(sd_radv_set_rdnss(ra
, 60, &test_rdnss
, 1) >= 0);
316 assert_se(sd_radv_set_dnssl(ra
, 60, (char **)test_dnssl
) >= 0);
318 for (i
= 0; i
< ELEMENTSOF(prefix
); i
++) {
321 printf("Test prefix %u\n", i
);
322 assert_se(sd_radv_prefix_new(&p
) >= 0);
324 assert_se(sd_radv_prefix_set_prefix(p
, &prefix
[i
].address
,
325 prefix
[i
].prefixlen
) >= 0);
327 assert_se(sd_radv_prefix_set_valid_lifetime(p
, prefix
[i
].valid
) >= 0);
328 if (prefix
[i
].preferred
)
329 assert_se(sd_radv_prefix_set_preferred_lifetime(p
, prefix
[i
].preferred
) >= 0);
331 assert_se((sd_radv_add_prefix(ra
, p
, false) >= 0) == prefix
[i
].succesful
);
332 assert_se(sd_radv_add_prefix(ra
, p
, false) < 0);
334 p
= sd_radv_prefix_unref(p
);
338 assert_se(sd_event_add_io(e
, &recv_router_advertisement
, test_fd
[0],
339 EPOLLIN
, radv_recv
, ra
) >= 0);
341 assert_se(sd_event_add_time(e
, &test_hangcheck
, clock_boottime_or_monotonic(),
342 time_now
+ 2 *USEC_PER_SEC
, 0,
343 test_rs_hangcheck
, NULL
) >= 0);
345 assert_se(sd_radv_start(ra
) >= 0);
349 test_hangcheck
= sd_event_source_unref(test_hangcheck
);
351 ra
= sd_radv_unref(ra
);
359 int main(int argc
, char *argv
[]) {
361 test_setup_logging(LOG_DEBUG
);