1 /* SPDX-License-Identifier: LGPL-2.1-or-later */
4 #include <netinet/icmp6.h>
10 #include "alloc-util.h"
13 #include "icmp6-packet.h"
14 #include "icmp6-util-unix.h"
15 #include "ndisc-internal.h"
16 #include "ndisc-option.h"
17 #include "socket-util.h"
19 static void test_with_sd_ndisc(const uint8_t *data
, size_t size
) {
20 struct ether_addr mac_addr
= {
21 .ether_addr_octet
= {'A', 'B', 'C', '1', '2', '3'}
23 _cleanup_(sd_event_unrefp
) sd_event
*e
= NULL
;
24 _cleanup_(sd_ndisc_unrefp
) sd_ndisc
*nd
= NULL
;
26 assert_se(sd_event_new(&e
) >= 0);
27 assert_se(sd_ndisc_new(&nd
) >= 0);
28 assert_se(sd_ndisc_attach_event(nd
, e
, 0) >= 0);
29 assert_se(sd_ndisc_set_ifindex(nd
, 42) >= 0);
30 assert_se(sd_ndisc_set_mac(nd
, &mac_addr
) >= 0);
31 assert_se(sd_ndisc_start(nd
) >= 0);
32 assert_se(write(test_fd
[1], data
, size
) == (ssize_t
) size
);
33 (void) sd_event_run(e
, UINT64_MAX
);
34 assert_se(sd_ndisc_stop(nd
) >= 0);
35 test_fd
[1] = safe_close(test_fd
[1]);
36 TAKE_FD(test_fd
[0]); /* It should be already closed by sd_ndisc_stop(). */
39 static void test_with_sd_radv(const uint8_t *data
, size_t size
) {
40 struct ether_addr mac_addr
= {
41 .ether_addr_octet
= {'A', 'B', 'C', '1', '2', '3'}
43 _cleanup_(sd_event_unrefp
) sd_event
*e
= NULL
;
44 _cleanup_(sd_radv_unrefp
) sd_radv
*ra
= NULL
;
46 assert_se(socketpair(AF_UNIX
, SOCK_SEQPACKET
| SOCK_CLOEXEC
| SOCK_NONBLOCK
, 0, test_fd
) >= 0);
48 assert_se(sd_event_new(&e
) >= 0);
49 assert_se(sd_radv_new(&ra
) >= 0);
50 assert_se(sd_radv_attach_event(ra
, e
, 0) >= 0);
51 assert_se(sd_radv_set_ifindex(ra
, 42) >= 0);
52 assert_se(sd_radv_set_mac(ra
, &mac_addr
) >= 0);
53 assert_se(sd_radv_start(ra
) >= 0);
54 assert_se(write(test_fd
[0], data
, size
) == (ssize_t
) size
);
55 (void) sd_event_run(e
, UINT64_MAX
);
56 assert_se(sd_radv_stop(ra
) >= 0);
57 test_fd
[0] = safe_close(test_fd
[0]);
58 TAKE_FD(test_fd
[1]); /* It should be already closed by sd_radv_stop(). */
61 static void test_with_icmp6_packet(const uint8_t *data
, size_t size
) {
62 _cleanup_close_pair_
int fd_pair
[2] = EBADF_PAIR
;
63 _cleanup_(icmp6_packet_unrefp
) ICMP6Packet
*packet
= NULL
;
64 _cleanup_set_free_ Set
*options
= NULL
;
66 assert_se(socketpair(AF_UNIX
, SOCK_DGRAM
| SOCK_CLOEXEC
| SOCK_NONBLOCK
, 0, fd_pair
) >= 0);
67 assert_se(write(fd_pair
[1], data
, size
) == (ssize_t
) size
);
69 if (icmp6_packet_receive(fd_pair
[0], &packet
) < 0)
72 if (ndisc_parse_options(packet
, &options
) < 0)
75 if (ndisc_send(fd_pair
[1], &IN6_ADDR_ALL_ROUTERS_MULTICAST
,
76 icmp6_packet_get_header(packet
), options
, /* timestamp = */ 0) < 0)
79 packet
= icmp6_packet_unref(packet
);
80 options
= set_free(options
);
82 if (icmp6_packet_receive(fd_pair
[0], &packet
) < 0)
85 assert_se(ndisc_parse_options(packet
, &options
) >= 0);
88 int LLVMFuzzerTestOneInput(const uint8_t *data
, size_t size
) {
89 if (outside_size_range(size
, 0, 2048))
94 test_with_sd_ndisc(data
, size
);
95 test_with_sd_radv(data
, size
);
96 test_with_icmp6_packet(data
, size
);