1 /* SPDX-License-Identifier: LGPL-2.1+ */
8 #include "networkd-link.h"
9 #include "networkd-lldp-rx.h"
10 #include "networkd-lldp-tx.h"
11 #include "networkd-network.h"
12 #include "string-table.h"
13 #include "string-util.h"
14 #include "tmpfile-util.h"
16 DEFINE_CONFIG_PARSE_ENUM(config_parse_lldp_mode
, lldp_mode
, LLDPMode
, "Failed to parse LLDP= setting.");
18 static const char* const lldp_mode_table
[_LLDP_MODE_MAX
] = {
19 [LLDP_MODE_NO
] = "no",
20 [LLDP_MODE_YES
] = "yes",
21 [LLDP_MODE_ROUTERS_ONLY
] = "routers-only",
24 DEFINE_STRING_TABLE_LOOKUP_WITH_BOOLEAN(lldp_mode
, LLDPMode
, LLDP_MODE_YES
);
26 bool link_lldp_rx_enabled(Link
*link
) {
29 if (link
->flags
& IFF_LOOPBACK
)
32 if (link
->iftype
!= ARPHRD_ETHER
)
38 /* LLDP should be handled on bridge slaves as those have a direct
39 * connection to their peers not on the bridge master. Linux doesn't
40 * even (by default) forward lldp packets to the bridge master.*/
41 if (streq_ptr("bridge", link
->kind
))
44 return link
->network
->lldp_mode
!= LLDP_MODE_NO
;
47 static void lldp_handler(sd_lldp
*lldp
, sd_lldp_event event
, sd_lldp_neighbor
*n
, void *userdata
) {
48 Link
*link
= userdata
;
53 (void) link_lldp_save(link
);
55 if (link_lldp_emit_enabled(link
) && event
== SD_LLDP_EVENT_ADDED
) {
56 /* If we received information about a new neighbor, restart the LLDP "fast" logic */
58 log_link_debug(link
, "Received LLDP datagram from previously unknown neighbor, restarting 'fast' LLDP transmission.");
60 r
= link_lldp_emit_start(link
);
62 log_link_warning_errno(link
, r
, "Failed to restart LLDP transmission: %m");
66 int link_lldp_rx_configure(Link
*link
) {
69 r
= sd_lldp_new(&link
->lldp
);
73 r
= sd_lldp_set_ifindex(link
->lldp
, link
->ifindex
);
77 r
= sd_lldp_match_capabilities(link
->lldp
,
78 link
->network
->lldp_mode
== LLDP_MODE_ROUTERS_ONLY
?
79 SD_LLDP_SYSTEM_CAPABILITIES_ALL_ROUTERS
:
80 SD_LLDP_SYSTEM_CAPABILITIES_ALL
);
84 r
= sd_lldp_set_filter_address(link
->lldp
, &link
->mac
);
88 r
= sd_lldp_attach_event(link
->lldp
, NULL
, 0);
92 r
= sd_lldp_set_callback(link
->lldp
, lldp_handler
, link
);
96 r
= link_update_lldp(link
);
103 int link_update_lldp(Link
*link
) {
111 if (link
->flags
& IFF_UP
) {
112 r
= sd_lldp_start(link
->lldp
);
114 return log_link_warning_errno(link
, r
, "Failed to start LLDP: %m");
116 log_link_debug(link
, "Started LLDP.");
118 r
= sd_lldp_stop(link
->lldp
);
120 return log_link_warning_errno(link
, r
, "Failed to stop LLDP: %m");
122 log_link_debug(link
, "Stopped LLDP.");
128 int link_lldp_save(Link
*link
) {
129 _cleanup_free_
char *temp_path
= NULL
;
130 _cleanup_fclose_
FILE *f
= NULL
;
131 sd_lldp_neighbor
**l
= NULL
;
135 assert(link
->lldp_file
);
138 (void) unlink(link
->lldp_file
);
142 r
= sd_lldp_get_neighbors(link
->lldp
, &l
);
146 (void) unlink(link
->lldp_file
);
152 r
= fopen_temporary(link
->lldp_file
, &f
, &temp_path
);
156 fchmod(fileno(f
), 0644);
158 for (i
= 0; i
< n
; i
++) {
163 r
= sd_lldp_neighbor_get_raw(l
[i
], &p
, &sz
);
168 (void) fwrite(&u
, 1, sizeof(u
), f
);
169 (void) fwrite(p
, 1, sz
, f
);
172 r
= fflush_and_check(f
);
176 if (rename(temp_path
, link
->lldp_file
) < 0) {
183 (void) unlink(link
->lldp_file
);
185 (void) unlink(temp_path
);
187 log_link_error_errno(link
, r
, "Failed to save LLDP data to %s: %m", link
->lldp_file
);
191 for (i
= 0; i
< n
; i
++)
192 sd_lldp_neighbor_unref(l
[i
]);