]> git.ipfire.org Git - thirdparty/systemd.git/blame - src/libsystemd-network/test-ipv4ll.c
util-lib: split our string related calls from util.[ch] into its own file string...
[thirdparty/systemd.git] / src / libsystemd-network / test-ipv4ll.c
CommitLineData
d9bf4f8c
UTL
1/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
2/***
3 This file is part of systemd.
4
5 Copyright (C) 2014 Axis Communications AB. All rights reserved.
6
7 systemd is free software; you can redistribute it and/or modify it
8 under the terms of the GNU Lesser General Public License as published by
9 the Free Software Foundation; either version 2.1 of the License, or
10 (at your option) any later version.
11
12 systemd is distributed in the hope that it will be useful, but
13 WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 Lesser General Public License for more details.
16
17 You should have received a copy of the GNU Lesser General Public License
18 along with systemd; If not, see <http://www.gnu.org/licenses/>.
19***/
20
d9bf4f8c
UTL
21#include <assert.h>
22#include <errno.h>
23#include <stdio.h>
07630cea 24#include <stdlib.h>
d9bf4f8c 25#include <sys/socket.h>
07630cea 26#include <sys/types.h>
d9bf4f8c
UTL
27#include <unistd.h>
28
d9bf4f8c 29#include "sd-ipv4ll.h"
07630cea 30
996d1697 31#include "arp-util.h"
07630cea
LP
32#include "event-util.h"
33#include "socket-util.h"
34#include "util.h"
d9bf4f8c
UTL
35
36static bool verbose = false;
37static bool extended = false;
38static int test_fd[2];
39
40static int basic_request_handler_bind = 0;
41static int basic_request_handler_stop = 0;
89ca10c6 42static void* basic_request_handler_userdata = (void*)0xCABCAB;
d9bf4f8c 43static void basic_request_handler(sd_ipv4ll *ll, int event, void *userdata) {
89ca10c6 44 assert_se(userdata == basic_request_handler_userdata);
d9bf4f8c
UTL
45
46 switch(event) {
be19c5b5 47 case SD_IPV4LL_EVENT_STOP:
d9bf4f8c
UTL
48 basic_request_handler_stop = 1;
49 break;
be19c5b5 50 case SD_IPV4LL_EVENT_BIND:
d9bf4f8c
UTL
51 basic_request_handler_bind = 1;
52 break;
53 default:
54 assert_se(0);
55 break;
56 }
57}
58
996d1697
TG
59static int arp_network_send_raw_socket(int fd, int ifindex,
60 const struct ether_arp *arp) {
d9bf4f8c 61 assert_se(arp);
996d1697 62 assert_se(ifindex > 0);
d9bf4f8c
UTL
63 assert_se(fd >= 0);
64
65 if (send(fd, arp, sizeof(struct ether_arp), 0) < 0)
66 return -errno;
67
68 return 0;
69}
70
996d1697
TG
71int arp_send_probe(int fd, int ifindex,
72 be32_t pa, const struct ether_addr *ha) {
73 struct ether_arp ea = {};
d9bf4f8c 74
996d1697
TG
75 assert(fd >= 0);
76 assert(ifindex > 0);
77 assert(pa != 0);
78 assert(ha);
d9bf4f8c 79
996d1697 80 return arp_network_send_raw_socket(fd, ifindex, &ea);
d9bf4f8c
UTL
81}
82
996d1697
TG
83int arp_send_announcement(int fd, int ifindex,
84 be32_t pa, const struct ether_addr *ha) {
85 struct ether_arp ea = {};
d9bf4f8c 86
996d1697
TG
87 assert(fd >= 0);
88 assert(ifindex > 0);
89 assert(pa != 0);
90 assert(ha);
d9bf4f8c 91
996d1697 92 return arp_network_send_raw_socket(fd, ifindex, &ea);
d9bf4f8c
UTL
93}
94
996d1697
TG
95int arp_network_bind_raw_socket(int index, be32_t address, const struct ether_addr *eth_mac) {
96 if (socketpair(AF_UNIX, SOCK_DGRAM | SOCK_NONBLOCK, 0, test_fd) < 0)
97 return -errno;
d9bf4f8c 98
996d1697 99 return test_fd[0];
d9bf4f8c
UTL
100}
101
102static void test_public_api_setters(sd_event *e) {
129dc1b4 103 struct in_addr address = {};
5625be76 104 unsigned seed = 0;
d9bf4f8c
UTL
105 sd_ipv4ll *ll;
106 struct ether_addr mac_addr = {
107 .ether_addr_octet = {'A', 'B', 'C', '1', '2', '3'}};
108
109 if (verbose)
110 printf("* %s\n", __FUNCTION__);
111
112 assert_se(sd_ipv4ll_new(&ll) == 0);
113 assert_se(ll);
114
115 assert_se(sd_ipv4ll_attach_event(NULL, NULL, 0) == -EINVAL);
116 assert_se(sd_ipv4ll_attach_event(ll, e, 0) == 0);
117 assert_se(sd_ipv4ll_attach_event(ll, e, 0) == -EBUSY);
118
119 assert_se(sd_ipv4ll_set_callback(NULL, NULL, NULL) == -EINVAL);
120 assert_se(sd_ipv4ll_set_callback(ll, NULL, NULL) == 0);
121
129dc1b4
TG
122 assert_se(sd_ipv4ll_set_address(ll, &address) == -EINVAL);
123 address.s_addr |= htobe32(169U << 24 | 254U << 16);
124 assert_se(sd_ipv4ll_set_address(ll, &address) == -EINVAL);
125 address.s_addr |= htobe32(0x00FF);
126 assert_se(sd_ipv4ll_set_address(ll, &address) == -EINVAL);
127 address.s_addr |= htobe32(0xF000);
128 assert_se(sd_ipv4ll_set_address(ll, &address) == 0);
129 address.s_addr |= htobe32(0x0F00);
130 assert_se(sd_ipv4ll_set_address(ll, &address) == -EINVAL);
131
5625be76
RC
132 assert_se(sd_ipv4ll_set_address_seed(NULL, seed) == -EINVAL);
133 assert_se(sd_ipv4ll_set_address_seed(ll, seed) == 0);
d9bf4f8c
UTL
134
135 assert_se(sd_ipv4ll_set_mac(NULL, NULL) == -EINVAL);
136 assert_se(sd_ipv4ll_set_mac(ll, NULL) == -EINVAL);
137 assert_se(sd_ipv4ll_set_mac(ll, &mac_addr) == 0);
138
139 assert_se(sd_ipv4ll_set_index(NULL, -1) == -EINVAL);
140 assert_se(sd_ipv4ll_set_index(ll, -1) == -EINVAL);
141 assert_se(sd_ipv4ll_set_index(ll, -99) == -EINVAL);
142 assert_se(sd_ipv4ll_set_index(ll, 1) == 0);
143 assert_se(sd_ipv4ll_set_index(ll, 99) == 0);
144
145 assert_se(sd_ipv4ll_ref(ll) == ll);
b45e4eb6 146 assert_se(sd_ipv4ll_unref(ll) == NULL);
d9bf4f8c
UTL
147
148 /* Cleanup */
149 assert_se(sd_ipv4ll_unref(ll) == NULL);
150}
151
152static void test_basic_request(sd_event *e) {
153
154 sd_ipv4ll *ll;
155 struct ether_arp arp;
156 struct ether_addr mac_addr = {
157 .ether_addr_octet = {'A', 'B', 'C', '1', '2', '3'}};
158
159 if (verbose)
160 printf("* %s\n", __FUNCTION__);
161
162 assert_se(sd_ipv4ll_new(&ll) == 0);
163 assert_se(sd_ipv4ll_start(ll) == -EINVAL);
164
165 assert_se(sd_ipv4ll_attach_event(ll, e, 0) == 0);
166 assert_se(sd_ipv4ll_start(ll) == -EINVAL);
167
168 assert_se(sd_ipv4ll_set_mac(ll, &mac_addr) == 0);
169 assert_se(sd_ipv4ll_start(ll) == -EINVAL);
170
171 assert_se(sd_ipv4ll_set_callback(ll, basic_request_handler,
89ca10c6 172 basic_request_handler_userdata) == 0);
d9bf4f8c
UTL
173 assert_se(sd_ipv4ll_start(ll) == -EINVAL);
174
175 assert_se(sd_ipv4ll_set_index(ll, 1) == 0);
176 assert_se(sd_ipv4ll_start(ll) == 0);
177
178 sd_event_run(e, (uint64_t) -1);
179 assert_se(sd_ipv4ll_start(ll) == -EBUSY);
180
e3dca008
TG
181 assert_se(sd_ipv4ll_is_running(ll));
182
d9bf4f8c
UTL
183 /* PROBE */
184 sd_event_run(e, (uint64_t) -1);
185 assert_se(read(test_fd[1], &arp, sizeof(struct ether_arp)) == sizeof(struct ether_arp));
d9bf4f8c
UTL
186
187 if (extended) {
188 /* PROBE */
189 sd_event_run(e, (uint64_t) -1);
190 assert_se(read(test_fd[1], &arp, sizeof(struct ether_arp)) == sizeof(struct ether_arp));
d9bf4f8c
UTL
191
192 /* PROBE */
193 sd_event_run(e, (uint64_t) -1);
194 assert_se(read(test_fd[1], &arp, sizeof(struct ether_arp)) == sizeof(struct ether_arp));
d9bf4f8c
UTL
195
196 sd_event_run(e, (uint64_t) -1);
197 assert_se(basic_request_handler_bind == 1);
198 }
199
200 sd_ipv4ll_stop(ll);
201 assert_se(basic_request_handler_stop == 1);
202
203 /* Cleanup */
204 assert_se(sd_ipv4ll_unref(ll) == NULL);
205 safe_close(test_fd[1]);
206}
207
208int main(int argc, char *argv[]) {
a64edefa 209 _cleanup_event_unref_ sd_event *e = NULL;
d9bf4f8c 210
e3dca008
TG
211 log_set_max_level(LOG_DEBUG);
212 log_parse_environment();
213 log_open();
214
d9bf4f8c
UTL
215 assert_se(sd_event_new(&e) >= 0);
216
217 test_public_api_setters(e);
d9bf4f8c
UTL
218 test_basic_request(e);
219
220 return 0;
221}