1 /* SPDX-License-Identifier: LGPL-2.1+ */
5 #include <net/ethernet.h>
13 #include "alloc-util.h"
15 #include "lldp-network.h"
17 #include "string-util.h"
19 #define TEST_LLDP_PORT "em1"
20 #define TEST_LLDP_TYPE_SYSTEM_NAME "systemd-lldp"
21 #define TEST_LLDP_TYPE_SYSTEM_DESC "systemd-lldp-desc"
23 static int test_fd
[2] = { -1, -1 };
24 static int lldp_handler_calls
;
26 int lldp_network_bind_raw_socket(int ifindex
) {
27 if (socketpair(AF_UNIX
, SOCK_DGRAM
| SOCK_CLOEXEC
| SOCK_NONBLOCK
, 0, test_fd
) < 0)
33 static void lldp_handler(sd_lldp
*lldp
, sd_lldp_event event
, sd_lldp_neighbor
*n
, void *userdata
) {
37 static int start_lldp(sd_lldp
**lldp
, sd_event
*e
, sd_lldp_callback_t cb
, void *cb_data
) {
40 r
= sd_lldp_new(lldp
);
44 r
= sd_lldp_set_ifindex(*lldp
, 42);
48 r
= sd_lldp_set_callback(*lldp
, cb
, cb_data
);
52 r
= sd_lldp_attach_event(*lldp
, e
, 0);
56 r
= sd_lldp_start(*lldp
);
63 static int stop_lldp(sd_lldp
*lldp
) {
66 r
= sd_lldp_stop(lldp
);
70 r
= sd_lldp_detach_event(lldp
);
75 safe_close(test_fd
[1]);
80 static void test_receive_basic_packet(sd_event
*e
) {
82 static const uint8_t frame
[] = {
84 0x01, 0x80, 0xc2, 0x00, 0x00, 0x03, /* Destination MAC */
85 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, /* Source MAC */
86 0x88, 0xcc, /* Ethertype */
87 /* LLDP mandatory TLVs */
88 0x02, 0x07, 0x04, 0x00, 0x01, 0x02, /* Chassis: MAC, 00:01:02:03:04:05 */
90 0x04, 0x04, 0x05, 0x31, 0x2f, 0x33, /* Port: interface name, "1/3" */
91 0x06, 0x02, 0x00, 0x78, /* TTL: 120 seconds */
92 /* LLDP optional TLVs */
93 0x08, 0x04, 0x50, 0x6f, 0x72, 0x74, /* Port Description: "Port" */
94 0x0a, 0x03, 0x53, 0x59, 0x53, /* System Name: "SYS" */
95 0x0c, 0x04, 0x66, 0x6f, 0x6f, 0x00, /* System Description: "foo" (NULL-terminated) */
96 0x00, 0x00 /* End Of LLDPDU */
100 sd_lldp_neighbor
**neighbors
;
107 lldp_handler_calls
= 0;
108 assert_se(start_lldp(&lldp
, e
, lldp_handler
, NULL
) == 0);
110 assert_se(write(test_fd
[1], frame
, sizeof(frame
)) == sizeof(frame
));
112 assert_se(lldp_handler_calls
== 1);
113 assert_se(sd_lldp_get_neighbors(lldp
, &neighbors
) == 1);
115 assert_se(sd_lldp_neighbor_get_chassis_id(neighbors
[0], &type
, &data
, &length
) == 0);
116 assert_se(type
== SD_LLDP_CHASSIS_SUBTYPE_MAC_ADDRESS
);
117 assert_se(length
== ETH_ALEN
);
118 assert_se(!memcmp(data
, "\x00\x01\x02\x03\x04\x05", ETH_ALEN
));
120 assert_se(sd_lldp_neighbor_get_port_id(neighbors
[0], &type
, &data
, &length
) == 0);
121 assert_se(type
== SD_LLDP_PORT_SUBTYPE_INTERFACE_NAME
);
122 assert_se(length
== 3);
123 assert_se(!memcmp(data
, "1/3", 3));
125 assert_se(sd_lldp_neighbor_get_port_description(neighbors
[0], &str
) == 0);
126 assert_se(streq(str
, "Port"));
128 assert_se(sd_lldp_neighbor_get_system_name(neighbors
[0], &str
) == 0);
129 assert_se(streq(str
, "SYS"));
131 assert_se(sd_lldp_neighbor_get_system_description(neighbors
[0], &str
) == 0);
132 assert_se(streq(str
, "foo"));
134 assert_se(sd_lldp_neighbor_get_ttl(neighbors
[0], &ttl
) == 0);
135 assert_se(ttl
== 120);
137 sd_lldp_neighbor_unref(neighbors
[0]);
140 assert_se(stop_lldp(lldp
) == 0);
143 static void test_receive_incomplete_packet(sd_event
*e
) {
145 sd_lldp_neighbor
**neighbors
;
147 /* Ethernet header */
148 0x01, 0x80, 0xc2, 0x00, 0x00, 0x03, /* Destination MAC */
149 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, /* Source MAC */
150 0x88, 0xcc, /* Ethertype */
151 /* LLDP mandatory TLVs */
152 0x02, 0x07, 0x04, 0x00, 0x01, 0x02, /* Chassis: MAC, 00:01:02:03:04:05 */
154 0x04, 0x04, 0x05, 0x31, 0x2f, 0x33, /* Port: interface name, "1/3" */
156 0x00, 0x00 /* End Of LLDPDU */
159 lldp_handler_calls
= 0;
160 assert_se(start_lldp(&lldp
, e
, lldp_handler
, NULL
) == 0);
162 assert_se(write(test_fd
[1], frame
, sizeof(frame
)) == sizeof(frame
));
164 assert_se(lldp_handler_calls
== 0);
165 assert_se(sd_lldp_get_neighbors(lldp
, &neighbors
) == 0);
167 assert_se(stop_lldp(lldp
) == 0);
170 static void test_receive_oui_packet(sd_event
*e
) {
172 sd_lldp_neighbor
**neighbors
;
174 /* Ethernet header */
175 0x01, 0x80, 0xc2, 0x00, 0x00, 0x03, /* Destination MAC */
176 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, /* Source MAC */
177 0x88, 0xcc, /* Ethertype */
178 /* LLDP mandatory TLVs */
179 0x02, 0x07, 0x04, 0x00, 0x01, 0x02, /* Chassis: MAC, 00:01:02:03:04:05 */
181 0x04, 0x04, 0x05, 0x31, 0x2f, 0x33, /* Port TLV: interface name, "1/3" */
182 0x06, 0x02, 0x00, 0x78, /* TTL: 120 seconds */
183 /* LLDP optional TLVs */
184 0xfe, 0x06, 0x00, 0x80, 0xc2, 0x01, /* Port VLAN ID: 0x1234 */
186 0xfe, 0x07, 0x00, 0x80, 0xc2, 0x02, /* Port and protocol: flag 1, PPVID 0x7788 */
188 0xfe, 0x0d, 0x00, 0x80, 0xc2, 0x03, /* VLAN Name: ID 0x1234, name "Vlan51" */
189 0x12, 0x34, 0x06, 0x56, 0x6c, 0x61,
191 0xfe, 0x06, 0x00, 0x80, 0xc2, 0x06, /* Management VID: 0x0102 */
193 0xfe, 0x09, 0x00, 0x80, 0xc2, 0x07, /* Link aggregation: status 1, ID 0x00140012 */
194 0x01, 0x00, 0x14, 0x00, 0x12,
195 0xfe, 0x07, 0x00, 0x12, 0x0f, 0x02, /* 802.3 Power via MDI: PSE, MDI enabled */
197 0x00, 0x00 /* End of LLDPDU */
200 lldp_handler_calls
= 0;
201 assert_se(start_lldp(&lldp
, e
, lldp_handler
, NULL
) == 0);
203 assert_se(write(test_fd
[1], frame
, sizeof(frame
)) == sizeof(frame
));
205 assert_se(lldp_handler_calls
== 1);
206 assert_se(sd_lldp_get_neighbors(lldp
, &neighbors
) == 1);
208 assert_se(sd_lldp_neighbor_tlv_rewind(neighbors
[0]) >= 0);
209 assert_se(sd_lldp_neighbor_tlv_is_type(neighbors
[0], SD_LLDP_TYPE_CHASSIS_ID
) > 0);
210 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
211 assert_se(sd_lldp_neighbor_tlv_is_type(neighbors
[0], SD_LLDP_TYPE_PORT_ID
) > 0);
212 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
213 assert_se(sd_lldp_neighbor_tlv_is_type(neighbors
[0], SD_LLDP_TYPE_TTL
) > 0);
214 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
215 assert_se(sd_lldp_neighbor_tlv_is_oui(neighbors
[0], SD_LLDP_OUI_802_1
, SD_LLDP_OUI_802_1_SUBTYPE_PORT_VLAN_ID
) > 0);
216 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
217 assert_se(sd_lldp_neighbor_tlv_is_oui(neighbors
[0], SD_LLDP_OUI_802_1
, SD_LLDP_OUI_802_1_SUBTYPE_PORT_PROTOCOL_VLAN_ID
) > 0);
218 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
219 assert_se(sd_lldp_neighbor_tlv_is_oui(neighbors
[0], SD_LLDP_OUI_802_1
, SD_LLDP_OUI_802_1_SUBTYPE_VLAN_NAME
) > 0);
220 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
221 assert_se(sd_lldp_neighbor_tlv_is_oui(neighbors
[0], SD_LLDP_OUI_802_1
, SD_LLDP_OUI_802_1_SUBTYPE_MANAGEMENT_VID
) > 0);
222 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
223 assert_se(sd_lldp_neighbor_tlv_is_oui(neighbors
[0], SD_LLDP_OUI_802_1
, SD_LLDP_OUI_802_1_SUBTYPE_LINK_AGGREGATION
) > 0);
224 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
225 assert_se(sd_lldp_neighbor_tlv_is_oui(neighbors
[0], SD_LLDP_OUI_802_3
, SD_LLDP_OUI_802_3_SUBTYPE_POWER_VIA_MDI
) > 0);
226 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
227 assert_se(sd_lldp_neighbor_tlv_is_type(neighbors
[0], SD_LLDP_TYPE_END
) > 0);
228 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) == 0);
230 sd_lldp_neighbor_unref(neighbors
[0]);
233 assert_se(stop_lldp(lldp
) == 0);
236 static void test_multiple_neighbors_sorted(sd_event
*e
) {
238 static const uint8_t frame1
[] = {
239 /* Ethernet header */
240 0x01, 0x80, 0xc2, 0x00, 0x00, 0x03, /* Destination MAC */
241 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, /* Source MAC */
242 0x88, 0xcc, /* Ethertype */
243 /* LLDP mandatory TLVs */
244 0x02, 0x04, 0x01, '1', '/', '2', /* Chassis component: "1/2" */
245 0x04, 0x04, 0x02, '2', '/', '3', /* Port component: "2/3" */
246 0x06, 0x02, 0x00, 0x78, /* TTL: 120 seconds */
247 0x00, 0x00 /* End Of LLDPDU */
249 static const uint8_t frame2
[] = {
250 /* Ethernet header */
251 0x01, 0x80, 0xc2, 0x00, 0x00, 0x03, /* Destination MAC */
252 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, /* Source MAC */
253 0x88, 0xcc, /* Ethertype */
254 /* LLDP mandatory TLVs */
255 0x02, 0x04, 0x01, '2', '/', '1', /* Chassis component: "2/1" */
256 0x04, 0x04, 0x02, '1', '/', '3', /* Port component: "1/3" */
257 0x06, 0x02, 0x00, 0x78, /* TTL: 120 seconds */
258 0x00, 0x00 /* End Of LLDPDU */
260 static const uint8_t frame3
[] = {
261 /* Ethernet header */
262 0x01, 0x80, 0xc2, 0x00, 0x00, 0x03, /* Destination MAC */
263 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, /* Source MAC */
264 0x88, 0xcc, /* Ethertype */
265 /* LLDP mandatory TLVs */
266 0x02, 0x05, 0x01, '2', '/', '1', '0', /* Chassis component: "2/10" */
267 0x04, 0x04, 0x02, '1', '/', '0', /* Port component: "1/0" */
268 0x06, 0x02, 0x00, 0x78, /* TTL: 120 seconds */
269 0x00, 0x00 /* End Of LLDPDU */
271 static const uint8_t frame4
[] = {
272 /* Ethernet header */
273 0x01, 0x80, 0xc2, 0x00, 0x00, 0x03, /* Destination MAC */
274 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, /* Source MAC */
275 0x88, 0xcc, /* Ethertype */
276 /* LLDP mandatory TLVs */
277 0x02, 0x05, 0x01, '2', '/', '1', '9', /* Chassis component: "2/19" */
278 0x04, 0x04, 0x02, '1', '/', '0', /* Port component: "1/0" */
279 0x06, 0x02, 0x00, 0x78, /* TTL: 120 seconds */
280 0x00, 0x00 /* End Of LLDPDU */
282 static const uint8_t frame5
[] = {
283 /* Ethernet header */
284 0x01, 0x80, 0xc2, 0x00, 0x00, 0x03, /* Destination MAC */
285 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, /* Source MAC */
286 0x88, 0xcc, /* Ethertype */
287 /* LLDP mandatory TLVs */
288 0x02, 0x04, 0x01, '1', '/', '2', /* Chassis component: "1/2" */
289 0x04, 0x05, 0x02, '2', '/', '1', '0', /* Port component: "2/10" */
290 0x06, 0x02, 0x00, 0x78, /* TTL: 120 seconds */
291 0x00, 0x00 /* End Of LLDPDU */
293 static const uint8_t frame6
[] = {
294 /* Ethernet header */
295 0x01, 0x80, 0xc2, 0x00, 0x00, 0x03, /* Destination MAC */
296 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, /* Source MAC */
297 0x88, 0xcc, /* Ethertype */
298 /* LLDP mandatory TLVs */
299 0x02, 0x04, 0x01, '1', '/', '2', /* Chassis component: "1/2" */
300 0x04, 0x05, 0x02, '2', '/', '3', '9', /* Port component: "2/10" */
301 0x06, 0x02, 0x00, 0x78, /* TTL: 120 seconds */
302 0x00, 0x00 /* End Of LLDPDU */
304 static const char* expected
[] = {
305 /* ordered pairs of Chassis+Port */
315 sd_lldp_neighbor
**neighbors
;
319 size_t length
, expected_length
;
322 lldp_handler_calls
= 0;
323 assert_se(start_lldp(&lldp
, e
, lldp_handler
, NULL
) == 0);
325 assert_se(write(test_fd
[1], frame1
, sizeof(frame1
)) == sizeof(frame1
));
327 assert_se(write(test_fd
[1], frame2
, sizeof(frame2
)) == sizeof(frame2
));
329 assert_se(write(test_fd
[1], frame3
, sizeof(frame3
)) == sizeof(frame3
));
331 assert_se(write(test_fd
[1], frame4
, sizeof(frame4
)) == sizeof(frame4
));
333 assert_se(write(test_fd
[1], frame5
, sizeof(frame5
)) == sizeof(frame5
));
335 assert_se(write(test_fd
[1], frame6
, sizeof(frame6
)) == sizeof(frame6
));
337 assert_se(lldp_handler_calls
== 6);
339 assert_se(sd_lldp_get_neighbors(lldp
, &neighbors
) == 6);
341 for (i
= 0; i
< 6; i
++) {
342 assert_se(sd_lldp_neighbor_get_chassis_id(neighbors
[i
], &type
, &data
, &length
) == 0);
343 assert_se(type
== SD_LLDP_CHASSIS_SUBTYPE_CHASSIS_COMPONENT
);
344 expected_length
= strlen(expected
[2 * i
]);
345 assert_se(length
== expected_length
);
346 assert_se(memcmp(data
, expected
[2 * i
], expected_length
) == 0);
348 assert_se(sd_lldp_neighbor_get_port_id(neighbors
[i
], &type
, &data
, &length
) == 0);
349 assert_se(type
== SD_LLDP_PORT_SUBTYPE_PORT_COMPONENT
);
350 expected_length
= strlen(expected
[2 * i
+ 1]);
351 assert_se(length
== expected_length
);
352 assert_se(memcmp(data
, expected
[2 * i
+ 1], expected_length
) == 0);
354 assert_se(sd_lldp_neighbor_get_ttl(neighbors
[i
], &ttl
) == 0);
355 assert_se(ttl
== 120);
358 for (i
= 0; i
< 6; i
++)
359 sd_lldp_neighbor_unref(neighbors
[i
]);
362 assert_se(stop_lldp(lldp
) == 0);
365 int main(int argc
, char *argv
[]) {
366 _cleanup_(sd_event_unrefp
) sd_event
*e
= NULL
;
368 log_set_max_level(LOG_DEBUG
);
370 /* LLDP reception tests */
371 assert_se(sd_event_new(&e
) == 0);
372 test_receive_basic_packet(e
);
373 test_receive_incomplete_packet(e
);
374 test_receive_oui_packet(e
);
375 test_multiple_neighbors_sorted(e
);