1 /* SPDX-License-Identifier: LGPL-2.1+ */
3 This file is part of systemd.
5 Copyright (C) 2014 Tom Gundersen
6 Copyright (C) 2014 Susant Sahani
8 systemd is free software; you can redistribute it and/or modify it
9 under the terms of the GNU Lesser General Public License as published by
10 the Free Software Foundation; either version 2.1 of the License, or
11 (at your option) any later version.
13 systemd is distributed in the hope that it will be useful, but
14 WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 Lesser General Public License for more details.
18 You should have received a copy of the GNU Lesser General Public License
19 along with systemd; If not, see <http://www.gnu.org/licenses/>.
22 #include <arpa/inet.h>
24 #include <net/ethernet.h>
32 #include "alloc-util.h"
34 #include "lldp-network.h"
36 #include "string-util.h"
38 #define TEST_LLDP_PORT "em1"
39 #define TEST_LLDP_TYPE_SYSTEM_NAME "systemd-lldp"
40 #define TEST_LLDP_TYPE_SYSTEM_DESC "systemd-lldp-desc"
42 static int test_fd
[2] = { -1, -1 };
43 static int lldp_handler_calls
;
45 int lldp_network_bind_raw_socket(int ifindex
) {
46 if (socketpair(AF_UNIX
, SOCK_DGRAM
| SOCK_NONBLOCK
, 0, test_fd
) < 0)
52 static void lldp_handler(sd_lldp
*lldp
, sd_lldp_event event
, sd_lldp_neighbor
*n
, void *userdata
) {
56 static int start_lldp(sd_lldp
**lldp
, sd_event
*e
, sd_lldp_callback_t cb
, void *cb_data
) {
59 r
= sd_lldp_new(lldp
);
63 r
= sd_lldp_set_ifindex(*lldp
, 42);
67 r
= sd_lldp_set_callback(*lldp
, cb
, cb_data
);
71 r
= sd_lldp_attach_event(*lldp
, e
, 0);
75 r
= sd_lldp_start(*lldp
);
82 static int stop_lldp(sd_lldp
*lldp
) {
85 r
= sd_lldp_stop(lldp
);
89 r
= sd_lldp_detach_event(lldp
);
94 safe_close(test_fd
[1]);
99 static void test_receive_basic_packet(sd_event
*e
) {
101 static const uint8_t frame
[] = {
102 /* Ethernet header */
103 0x01, 0x80, 0xc2, 0x00, 0x00, 0x03, /* Destination MAC */
104 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, /* Source MAC */
105 0x88, 0xcc, /* Ethertype */
106 /* LLDP mandatory TLVs */
107 0x02, 0x07, 0x04, 0x00, 0x01, 0x02, /* Chassis: MAC, 00:01:02:03:04:05 */
109 0x04, 0x04, 0x05, 0x31, 0x2f, 0x33, /* Port: interface name, "1/3" */
110 0x06, 0x02, 0x00, 0x78, /* TTL: 120 seconds */
111 /* LLDP optional TLVs */
112 0x08, 0x04, 0x50, 0x6f, 0x72, 0x74, /* Port Description: "Port" */
113 0x0a, 0x03, 0x53, 0x59, 0x53, /* System Name: "SYS" */
114 0x0c, 0x04, 0x66, 0x6f, 0x6f, 0x00, /* System Description: "foo" (NULL-terminated) */
115 0x00, 0x00 /* End Of LLDPDU */
119 sd_lldp_neighbor
**neighbors
;
126 lldp_handler_calls
= 0;
127 assert_se(start_lldp(&lldp
, e
, lldp_handler
, NULL
) == 0);
129 assert_se(write(test_fd
[1], frame
, sizeof(frame
)) == sizeof(frame
));
131 assert_se(lldp_handler_calls
== 1);
132 assert_se(sd_lldp_get_neighbors(lldp
, &neighbors
) == 1);
134 assert_se(sd_lldp_neighbor_get_chassis_id(neighbors
[0], &type
, &data
, &length
) == 0);
135 assert_se(type
== SD_LLDP_CHASSIS_SUBTYPE_MAC_ADDRESS
);
136 assert_se(length
== ETH_ALEN
);
137 assert_se(!memcmp(data
, "\x00\x01\x02\x03\x04\x05", ETH_ALEN
));
139 assert_se(sd_lldp_neighbor_get_port_id(neighbors
[0], &type
, &data
, &length
) == 0);
140 assert_se(type
== SD_LLDP_PORT_SUBTYPE_INTERFACE_NAME
);
141 assert_se(length
== 3);
142 assert_se(strneq((char *) data
, "1/3", 3));
144 assert_se(sd_lldp_neighbor_get_port_description(neighbors
[0], &str
) == 0);
145 assert_se(streq(str
, "Port"));
147 assert_se(sd_lldp_neighbor_get_system_name(neighbors
[0], &str
) == 0);
148 assert_se(streq(str
, "SYS"));
150 assert_se(sd_lldp_neighbor_get_system_description(neighbors
[0], &str
) == 0);
151 assert_se(streq(str
, "foo"));
153 assert_se(sd_lldp_neighbor_get_ttl(neighbors
[0], &ttl
) == 0);
154 assert_se(ttl
== 120);
156 sd_lldp_neighbor_unref(neighbors
[0]);
159 assert_se(stop_lldp(lldp
) == 0);
162 static void test_receive_incomplete_packet(sd_event
*e
) {
164 sd_lldp_neighbor
**neighbors
;
166 /* Ethernet header */
167 0x01, 0x80, 0xc2, 0x00, 0x00, 0x03, /* Destination MAC */
168 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, /* Source MAC */
169 0x88, 0xcc, /* Ethertype */
170 /* LLDP mandatory TLVs */
171 0x02, 0x07, 0x04, 0x00, 0x01, 0x02, /* Chassis: MAC, 00:01:02:03:04:05 */
173 0x04, 0x04, 0x05, 0x31, 0x2f, 0x33, /* Port: interface name, "1/3" */
175 0x00, 0x00 /* End Of LLDPDU */
178 lldp_handler_calls
= 0;
179 assert_se(start_lldp(&lldp
, e
, lldp_handler
, NULL
) == 0);
181 assert_se(write(test_fd
[1], frame
, sizeof(frame
)) == sizeof(frame
));
183 assert_se(lldp_handler_calls
== 0);
184 assert_se(sd_lldp_get_neighbors(lldp
, &neighbors
) == 0);
186 assert_se(stop_lldp(lldp
) == 0);
189 static void test_receive_oui_packet(sd_event
*e
) {
191 sd_lldp_neighbor
**neighbors
;
193 /* Ethernet header */
194 0x01, 0x80, 0xc2, 0x00, 0x00, 0x03, /* Destination MAC */
195 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, /* Source MAC */
196 0x88, 0xcc, /* Ethertype */
197 /* LLDP mandatory TLVs */
198 0x02, 0x07, 0x04, 0x00, 0x01, 0x02, /* Chassis: MAC, 00:01:02:03:04:05 */
200 0x04, 0x04, 0x05, 0x31, 0x2f, 0x33, /* Port TLV: interface name, "1/3" */
201 0x06, 0x02, 0x00, 0x78, /* TTL: 120 seconds */
202 /* LLDP optional TLVs */
203 0xfe, 0x06, 0x00, 0x80, 0xc2, 0x01, /* Port VLAN ID: 0x1234 */
205 0xfe, 0x07, 0x00, 0x80, 0xc2, 0x02, /* Port and protocol: flag 1, PPVID 0x7788 */
207 0xfe, 0x0d, 0x00, 0x80, 0xc2, 0x03, /* VLAN Name: ID 0x1234, name "Vlan51" */
208 0x12, 0x34, 0x06, 0x56, 0x6c, 0x61,
210 0xfe, 0x06, 0x00, 0x80, 0xc2, 0x06, /* Management VID: 0x0102 */
212 0xfe, 0x09, 0x00, 0x80, 0xc2, 0x07, /* Link aggregation: status 1, ID 0x00140012 */
213 0x01, 0x00, 0x14, 0x00, 0x12,
214 0x00, 0x00 /* End of LLDPDU */
217 lldp_handler_calls
= 0;
218 assert_se(start_lldp(&lldp
, e
, lldp_handler
, NULL
) == 0);
220 assert_se(write(test_fd
[1], frame
, sizeof(frame
)) == sizeof(frame
));
222 assert_se(lldp_handler_calls
== 1);
223 assert_se(sd_lldp_get_neighbors(lldp
, &neighbors
) == 1);
225 assert_se(sd_lldp_neighbor_tlv_rewind(neighbors
[0]) >= 0);
226 assert_se(sd_lldp_neighbor_tlv_is_type(neighbors
[0], SD_LLDP_TYPE_CHASSIS_ID
) > 0);
227 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
228 assert_se(sd_lldp_neighbor_tlv_is_type(neighbors
[0], SD_LLDP_TYPE_PORT_ID
) > 0);
229 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
230 assert_se(sd_lldp_neighbor_tlv_is_type(neighbors
[0], SD_LLDP_TYPE_TTL
) > 0);
231 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
232 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);
233 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
234 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);
235 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
236 assert_se(sd_lldp_neighbor_tlv_is_oui(neighbors
[0], SD_LLDP_OUI_802_1
, SD_LLDP_OUI_802_1_SUBTYPE_VLAN_NAME
) > 0);
237 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
238 assert_se(sd_lldp_neighbor_tlv_is_oui(neighbors
[0], SD_LLDP_OUI_802_1
, SD_LLDP_OUI_802_1_SUBTYPE_MANAGEMENT_VID
) > 0);
239 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
240 assert_se(sd_lldp_neighbor_tlv_is_oui(neighbors
[0], SD_LLDP_OUI_802_1
, SD_LLDP_OUI_802_1_SUBTYPE_LINK_AGGREGATION
) > 0);
241 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
242 assert_se(sd_lldp_neighbor_tlv_is_type(neighbors
[0], SD_LLDP_TYPE_END
) > 0);
243 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) == 0);
245 sd_lldp_neighbor_unref(neighbors
[0]);
248 assert_se(stop_lldp(lldp
) == 0);
251 int main(int argc
, char *argv
[]) {
252 _cleanup_(sd_event_unrefp
) sd_event
*e
= NULL
;
254 log_set_max_level(LOG_DEBUG
);
256 /* LLDP reception tests */
257 assert_se(sd_event_new(&e
) == 0);
258 test_receive_basic_packet(e
);
259 test_receive_incomplete_packet(e
);
260 test_receive_oui_packet(e
);