1 /* SPDX-License-Identifier: LGPL-2.1+ */
4 #include <net/if_arp.h>
9 #include "networkd-link.h"
10 #include "networkd-lldp-rx.h"
11 #include "networkd-lldp-tx.h"
12 #include "networkd-network.h"
13 #include "string-table.h"
14 #include "string-util.h"
15 #include "tmpfile-util.h"
17 DEFINE_CONFIG_PARSE_ENUM(config_parse_lldp_mode
, lldp_mode
, LLDPMode
, "Failed to parse LLDP= setting.");
19 static const char* const lldp_mode_table
[_LLDP_MODE_MAX
] = {
20 [LLDP_MODE_NO
] = "no",
21 [LLDP_MODE_YES
] = "yes",
22 [LLDP_MODE_ROUTERS_ONLY
] = "routers-only",
25 DEFINE_STRING_TABLE_LOOKUP_WITH_BOOLEAN(lldp_mode
, LLDPMode
, LLDP_MODE_YES
);
27 bool link_lldp_rx_enabled(Link
*link
) {
30 if (link
->flags
& IFF_LOOPBACK
)
33 if (link
->iftype
!= ARPHRD_ETHER
)
39 /* LLDP should be handled on bridge slaves as those have a direct
40 * connection to their peers not on the bridge master. Linux doesn't
41 * even (by default) forward lldp packets to the bridge master.*/
42 if (streq_ptr("bridge", link
->kind
))
45 return link
->network
->lldp_mode
!= LLDP_MODE_NO
;
48 static void lldp_handler(sd_lldp
*lldp
, sd_lldp_event event
, sd_lldp_neighbor
*n
, void *userdata
) {
49 Link
*link
= userdata
;
54 (void) link_lldp_save(link
);
56 if (link_lldp_emit_enabled(link
) && event
== SD_LLDP_EVENT_ADDED
) {
57 /* If we received information about a new neighbor, restart the LLDP "fast" logic */
59 log_link_debug(link
, "Received LLDP datagram from previously unknown neighbor, restarting 'fast' LLDP transmission.");
61 r
= link_lldp_emit_start(link
);
63 log_link_warning_errno(link
, r
, "Failed to restart LLDP transmission: %m");
67 int link_lldp_rx_configure(Link
*link
) {
70 r
= sd_lldp_new(&link
->lldp
);
74 r
= sd_lldp_set_ifindex(link
->lldp
, link
->ifindex
);
78 r
= sd_lldp_match_capabilities(link
->lldp
,
79 link
->network
->lldp_mode
== LLDP_MODE_ROUTERS_ONLY
?
80 SD_LLDP_SYSTEM_CAPABILITIES_ALL_ROUTERS
:
81 SD_LLDP_SYSTEM_CAPABILITIES_ALL
);
85 r
= sd_lldp_set_filter_address(link
->lldp
, &link
->mac
);
89 r
= sd_lldp_attach_event(link
->lldp
, NULL
, 0);
93 r
= sd_lldp_set_callback(link
->lldp
, lldp_handler
, link
);
97 r
= link_update_lldp(link
);
104 int link_update_lldp(Link
*link
) {
112 if (link
->flags
& IFF_UP
) {
113 r
= sd_lldp_start(link
->lldp
);
115 return log_link_warning_errno(link
, r
, "Failed to start LLDP: %m");
117 log_link_debug(link
, "Started LLDP.");
119 r
= sd_lldp_stop(link
->lldp
);
121 return log_link_warning_errno(link
, r
, "Failed to stop LLDP: %m");
123 log_link_debug(link
, "Stopped LLDP.");
129 int link_lldp_save(Link
*link
) {
130 _cleanup_free_
char *temp_path
= NULL
;
131 _cleanup_fclose_
FILE *f
= NULL
;
132 sd_lldp_neighbor
**l
= NULL
;
136 assert(link
->lldp_file
);
139 (void) unlink(link
->lldp_file
);
143 r
= sd_lldp_get_neighbors(link
->lldp
, &l
);
147 (void) unlink(link
->lldp_file
);
153 r
= fopen_temporary(link
->lldp_file
, &f
, &temp_path
);
157 fchmod(fileno(f
), 0644);
159 for (i
= 0; i
< n
; i
++) {
164 r
= sd_lldp_neighbor_get_raw(l
[i
], &p
, &sz
);
169 (void) fwrite(&u
, 1, sizeof(u
), f
);
170 (void) fwrite(p
, 1, sz
, f
);
173 r
= fflush_and_check(f
);
177 if (rename(temp_path
, link
->lldp_file
) < 0) {
184 (void) unlink(link
->lldp_file
);
186 (void) unlink(temp_path
);
188 log_link_error_errno(link
, r
, "Failed to save LLDP data to %s: %m", link
->lldp_file
);
192 for (i
= 0; i
< n
; i
++)
193 sd_lldp_neighbor_unref(l
[i
]);