2 This file is part of systemd.
4 Copyright (C) 2014 Tom Gundersen
5 Copyright (C) 2014 Susant Sahani
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.
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.
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/>.
21 #include <arpa/inet.h>
22 #include <net/ethernet.h>
30 #include "alloc-util.h"
32 #include "lldp-network.h"
34 #include "string-util.h"
36 #define TEST_LLDP_PORT "em1"
37 #define TEST_LLDP_TYPE_SYSTEM_NAME "systemd-lldp"
38 #define TEST_LLDP_TYPE_SYSTEM_DESC "systemd-lldp-desc"
40 static int test_fd
[2] = { -1, -1 };
41 static int lldp_handler_calls
;
43 int lldp_network_bind_raw_socket(int ifindex
) {
44 if (socketpair(AF_UNIX
, SOCK_DGRAM
| SOCK_NONBLOCK
, 0, test_fd
) < 0)
50 static void lldp_handler(sd_lldp
*lldp
, sd_lldp_event event
, sd_lldp_neighbor
*n
, void *userdata
) {
54 static int start_lldp(sd_lldp
**lldp
, sd_event
*e
, sd_lldp_callback_t cb
, void *cb_data
) {
57 r
= sd_lldp_new(lldp
, 42);
61 r
= sd_lldp_attach_event(*lldp
, e
, 0);
65 r
= sd_lldp_set_callback(*lldp
, cb
, cb_data
);
69 r
= sd_lldp_start(*lldp
);
76 static int stop_lldp(sd_lldp
*lldp
) {
79 r
= sd_lldp_stop(lldp
);
83 r
= sd_lldp_detach_event(lldp
);
88 safe_close(test_fd
[1]);
93 static void test_receive_basic_packet(sd_event
*e
) {
95 static const uint8_t frame
[] = {
97 0x01, 0x80, 0xc2, 0x00, 0x00, 0x03, /* Destination MAC*/
98 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, /* Source MAC */
99 0x88, 0xcc, /* Ethertype */
100 /* LLDP mandatory TLVs */
101 0x02, 0x07, 0x04, 0x00, 0x01, 0x02, /* Chassis: MAC, 00:01:02:03:04:05 */
103 0x04, 0x04, 0x05, 0x31, 0x2f, 0x33, /* Port: interface name, "1/3" */
104 0x06, 0x02, 0x00, 0x78, /* TTL: 120 seconds*/
105 /* LLDP optional TLVs */
106 0x08, 0x04, 0x50, 0x6f, 0x72, 0x74, /* Port Description: "Port" */
107 0x0a, 0x03, 0x53, 0x59, 0x53, /* System Name: "SYS" */
108 0x0c, 0x04, 0x66, 0x6f, 0x6f, 0x00, /* System Description: "foo" (NULL-terminated) */
109 0x00, 0x00 /* End Of LLDPDU */
113 sd_lldp_neighbor
**neighbors
;
120 lldp_handler_calls
= 0;
121 assert_se(start_lldp(&lldp
, e
, lldp_handler
, NULL
) == 0);
123 assert_se(write(test_fd
[1], frame
, sizeof(frame
)) == sizeof(frame
));
125 assert_se(lldp_handler_calls
== 1);
126 assert_se(sd_lldp_get_neighbors(lldp
, &neighbors
) == 1);
128 assert_se(sd_lldp_neighbor_get_chassis_id(neighbors
[0], &type
, &data
, &length
) == 0);
129 assert_se(type
== SD_LLDP_CHASSIS_SUBTYPE_MAC_ADDRESS
);
130 assert_se(length
== ETH_ALEN
);
131 assert_se(!memcmp(data
, "\x00\x01\x02\x03\x04\x05", ETH_ALEN
));
133 assert_se(sd_lldp_neighbor_get_port_id(neighbors
[0], &type
, &data
, &length
) == 0);
134 assert_se(type
== SD_LLDP_PORT_SUBTYPE_INTERFACE_NAME
);
135 assert_se(length
== 3);
136 assert_se(strneq((char *) data
, "1/3", 3));
138 assert_se(sd_lldp_neighbor_get_port_description(neighbors
[0], &str
) == 0);
139 assert_se(streq(str
, "Port"));
141 assert_se(sd_lldp_neighbor_get_system_name(neighbors
[0], &str
) == 0);
142 assert_se(streq(str
, "SYS"));
144 assert_se(sd_lldp_neighbor_get_system_description(neighbors
[0], &str
) == 0);
145 assert_se(streq(str
, "foo"));
147 assert_se(sd_lldp_neighbor_get_ttl(neighbors
[0], &ttl
) == 0);
148 assert_se(ttl
== 120);
150 sd_lldp_neighbor_unref(neighbors
[0]);
153 assert_se(stop_lldp(lldp
) == 0);
156 static void test_receive_incomplete_packet(sd_event
*e
) {
158 sd_lldp_neighbor
**neighbors
;
160 /* Ethernet header */
161 0x01, 0x80, 0xc2, 0x00, 0x00, 0x03, /* Destination MAC*/
162 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, /* Source MAC */
163 0x88, 0xcc, /* Ethertype */
164 /* LLDP mandatory TLVs */
165 0x02, 0x07, 0x04, 0x00, 0x01, 0x02, /* Chassis: MAC, 00:01:02:03:04:05 */
167 0x04, 0x04, 0x05, 0x31, 0x2f, 0x33, /* Port: interface name, "1/3" */
169 0x00, 0x00 /* End Of LLDPDU */
172 lldp_handler_calls
= 0;
173 assert_se(start_lldp(&lldp
, e
, lldp_handler
, NULL
) == 0);
175 assert_se(write(test_fd
[1], frame
, sizeof(frame
)) == sizeof(frame
));
177 assert_se(lldp_handler_calls
== 0);
178 assert_se(sd_lldp_get_neighbors(lldp
, &neighbors
) == 0);
180 assert_se(stop_lldp(lldp
) == 0);
183 static void test_receive_oui_packet(sd_event
*e
) {
185 sd_lldp_neighbor
**neighbors
;
187 /* Ethernet header */
188 0x01, 0x80, 0xc2, 0x00, 0x00, 0x03, /* Destination MAC*/
189 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, /* Source MAC */
190 0x88, 0xcc, /* Ethertype */
191 /* LLDP mandatory TLVs */
192 0x02, 0x07, 0x04, 0x00, 0x01, 0x02, /* Chassis: MAC, 00:01:02:03:04:05 */
194 0x04, 0x04, 0x05, 0x31, 0x2f, 0x33, /* Port TLV: interface name, "1/3" */
195 0x06, 0x02, 0x00, 0x78, /* TTL: 120 seconds*/
196 /* LLDP optional TLVs */
197 0xfe, 0x06, 0x00, 0x80, 0xc2, 0x01, /* Port VLAN ID: 0x1234 */
199 0xfe, 0x07, 0x00, 0x80, 0xc2, 0x02, /* Port and protocol: flag 1, PPVID 0x7788 */
201 0xfe, 0x0d, 0x00, 0x80, 0xc2, 0x03, /* VLAN Name: ID 0x1234, name "Vlan51" */
202 0x12, 0x34, 0x06, 0x56, 0x6c, 0x61,
204 0xfe, 0x06, 0x00, 0x80, 0xc2, 0x06, /* Management VID: 0x0102 */
206 0xfe, 0x09, 0x00, 0x80, 0xc2, 0x07, /* Link aggregation: status 1, ID 0x00140012 */
207 0x01, 0x00, 0x14, 0x00, 0x12,
208 0x00, 0x00 /* End of LLDPDU */
211 lldp_handler_calls
= 0;
212 assert_se(start_lldp(&lldp
, e
, lldp_handler
, NULL
) == 0);
214 assert_se(write(test_fd
[1], frame
, sizeof(frame
)) == sizeof(frame
));
216 assert_se(lldp_handler_calls
== 1);
217 assert_se(sd_lldp_get_neighbors(lldp
, &neighbors
) == 1);
219 assert_se(sd_lldp_neighbor_tlv_rewind(neighbors
[0]) >= 0);
220 assert_se(sd_lldp_neighbor_tlv_is_type(neighbors
[0], SD_LLDP_TYPE_CHASSIS_ID
) > 0);
221 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
222 assert_se(sd_lldp_neighbor_tlv_is_type(neighbors
[0], SD_LLDP_TYPE_PORT_ID
) > 0);
223 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
224 assert_se(sd_lldp_neighbor_tlv_is_type(neighbors
[0], SD_LLDP_TYPE_TTL
) > 0);
225 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
226 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);
227 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
228 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);
229 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
230 assert_se(sd_lldp_neighbor_tlv_is_oui(neighbors
[0], SD_LLDP_OUI_802_1
, SD_LLDP_OUI_802_1_SUBTYPE_VLAN_NAME
) > 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_MANAGEMENT_VID
) > 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_LINK_AGGREGATION
) > 0);
235 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) > 0);
236 assert_se(sd_lldp_neighbor_tlv_is_type(neighbors
[0], SD_LLDP_TYPE_END
) > 0);
237 assert_se(sd_lldp_neighbor_tlv_next(neighbors
[0]) == 0);
239 sd_lldp_neighbor_unref(neighbors
[0]);
242 assert_se(stop_lldp(lldp
) == 0);
245 int main(int argc
, char *argv
[]) {
246 _cleanup_(sd_event_unrefp
) sd_event
*e
= NULL
;
248 log_set_max_level(LOG_DEBUG
);
250 /* LLDP reception tests */
251 assert_se(sd_event_new(&e
) == 0);
252 test_receive_basic_packet(e
);
253 test_receive_incomplete_packet(e
);
254 test_receive_oui_packet(e
);