• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

systemd / systemd / 14458263136

14 Apr 2025 06:41PM UTC coverage: 72.031% (+0.001%) from 72.03%
14458263136

push

github

yuwata
test: drop error conditions for old kernels (<3.2)

Now our baseline on the kernel is 5.4.

2 of 4 new or added lines in 1 file covered. (50.0%)

1428 existing lines in 44 files now uncovered.

297292 of 412726 relevant lines covered (72.03%)

683119.61 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

76.58
/src/shared/loopback-setup.c
1
/* SPDX-License-Identifier: LGPL-2.1-or-later */
2

3
#include <net/if.h>
4
#include <stdlib.h>
5

6
#include "sd-netlink.h"
7

8
#include "loopback-setup.h"
9
#include "missing_network.h"
10
#include "netlink-util.h"
11
#include "time-util.h"
12

13
#define LOOPBACK_SETUP_TIMEOUT_USEC (5 * USEC_PER_SEC)
14

15
struct state {
16
        unsigned n_messages;
17
        int rcode;
18
        const char *error_message;
19
        const char *success_message;
20
        const char *eexist_message;
21
};
22

23
static int generic_handler(sd_netlink *rtnl, sd_netlink_message *m, void *userdata) {
420✔
24
        struct state *s = ASSERT_PTR(userdata);
420✔
25
        int r;
420✔
26

27
        assert(s->n_messages > 0);
420✔
28
        s->n_messages--;
420✔
29

30
        errno = 0;
420✔
31

32
        r = sd_netlink_message_get_errno(m);
420✔
33
        if (r == -EEXIST && s->eexist_message)
420✔
34
                log_debug_errno(r, "%s", s->eexist_message);
54✔
35
        else if (r < 0)
366✔
36
                log_debug_errno(r, "%s: %m", s->error_message);
×
37
        else
38
                log_debug("%s", s->success_message);
366✔
39

40
        s->rcode = r;
420✔
41
        return 0;
420✔
42
}
43

44
static int start_loopback(sd_netlink *rtnl, struct state *s) {
140✔
45
        _cleanup_(sd_netlink_message_unrefp) sd_netlink_message *req = NULL;
140✔
46
        int r;
140✔
47

48
        assert(rtnl);
140✔
49
        assert(s);
140✔
50

51
        r = sd_rtnl_message_new_link(rtnl, &req, RTM_SETLINK, LOOPBACK_IFINDEX);
140✔
52
        if (r < 0)
140✔
53
                return r;
54

55
        r = sd_rtnl_message_link_set_flags(req, IFF_UP, IFF_UP);
140✔
56
        if (r < 0)
140✔
57
                return r;
58

59
        r = sd_netlink_call_async(rtnl, NULL, req, generic_handler, NULL, s, LOOPBACK_SETUP_TIMEOUT_USEC, "systemd-start-loopback");
140✔
60
        if (r < 0)
140✔
61
                return r;
62

63
        s->n_messages++;
140✔
64
        return 0;
140✔
65
}
66

67
static int add_ipv4_address(sd_netlink *rtnl, struct state *s) {
140✔
68
        _cleanup_(sd_netlink_message_unrefp) sd_netlink_message *req = NULL;
140✔
69
        int r;
140✔
70

71
        assert(rtnl);
140✔
72
        assert(s);
140✔
73

74
        r = sd_rtnl_message_new_addr(rtnl, &req, RTM_NEWADDR, LOOPBACK_IFINDEX, AF_INET);
140✔
75
        if (r < 0)
140✔
76
                return r;
77

78
        r = sd_rtnl_message_addr_set_prefixlen(req, 8);
140✔
79
        if (r < 0)
140✔
80
                return r;
81

82
        r = sd_rtnl_message_addr_set_scope(req, RT_SCOPE_HOST);
140✔
83
        if (r < 0)
140✔
84
                return r;
85

86
        r = sd_netlink_message_append_u32(req, IFA_FLAGS, IFA_F_PERMANENT);
140✔
87
        if (r < 0)
140✔
88
                return r;
89

90
        r = sd_netlink_message_append_in_addr(req, IFA_LOCAL, &(struct in_addr) { .s_addr = htobe32(INADDR_LOOPBACK) } );
140✔
91
        if (r < 0)
140✔
92
                return r;
93

94
        r = sd_netlink_call_async(rtnl, NULL, req, generic_handler, NULL, s, USEC_INFINITY, "systemd-loopback-ipv4");
140✔
95
        if (r < 0)
140✔
96
                return r;
97

98
        s->n_messages++;
140✔
99
        return 0;
140✔
100
}
101

102
static int add_ipv6_address(sd_netlink *rtnl, struct state *s) {
140✔
103
        _cleanup_(sd_netlink_message_unrefp) sd_netlink_message *req = NULL;
140✔
104
        int r;
140✔
105

106
        assert(rtnl);
140✔
107
        assert(s);
140✔
108

109
        r = sd_rtnl_message_new_addr(rtnl, &req, RTM_NEWADDR, LOOPBACK_IFINDEX, AF_INET6);
140✔
110
        if (r < 0)
140✔
111
                return r;
112

113
        r = sd_rtnl_message_addr_set_prefixlen(req, 128);
140✔
114
        if (r < 0)
140✔
115
                return r;
116

117
        uint32_t flags = IFA_F_PERMANENT|IFA_F_NOPREFIXROUTE;
140✔
118
        r = sd_netlink_message_append_u32(req, IFA_FLAGS, flags);
140✔
119
        if (r < 0)
140✔
120
                return r;
121

122
        r = sd_rtnl_message_addr_set_scope(req, RT_SCOPE_HOST);
140✔
123
        if (r < 0)
140✔
124
                return r;
125

126
        r = sd_netlink_message_append_in6_addr(req, IFA_LOCAL, &in6addr_loopback);
140✔
127
        if (r < 0)
140✔
128
                return r;
129

130
        r = sd_netlink_call_async(rtnl, NULL, req, generic_handler, NULL, s, USEC_INFINITY, "systemd-loopback-ipv6");
140✔
131
        if (r < 0)
140✔
132
                return r;
133

134
        s->n_messages++;
140✔
135
        return 0;
140✔
136
}
137

UNCOV
138
static int check_loopback(sd_netlink *rtnl) {
×
UNCOV
139
        _cleanup_(sd_netlink_message_unrefp) sd_netlink_message *req = NULL, *reply = NULL;
×
UNCOV
140
        unsigned flags;
×
UNCOV
141
        int r;
×
142

143
        r = sd_rtnl_message_new_link(rtnl, &req, RTM_GETLINK, LOOPBACK_IFINDEX);
×
144
        if (r < 0)
×
145
                return r;
146

UNCOV
147
        r = sd_netlink_call(rtnl, req, USEC_INFINITY, &reply);
×
148
        if (r < 0)
×
149
                return r;
150

UNCOV
151
        r = sd_rtnl_message_link_get_flags(reply, &flags);
×
152
        if (r < 0)
×
153
                return r;
154

UNCOV
155
        return flags & IFF_UP;
×
156
}
157

158
int loopback_setup(void) {
140✔
159
        _cleanup_(sd_netlink_unrefp) sd_netlink *rtnl = NULL;
140✔
160
        struct state state_4 = {
140✔
161
                .error_message = "Failed to add address 127.0.0.1 to loopback interface",
162
                .success_message = "Successfully added address 127.0.0.1 to loopback interface",
163
                .eexist_message = "127.0.0.1 has already been added to loopback interface",
164
        }, state_6 = {
140✔
165
                .error_message = "Failed to add address ::1 to loopback interface",
166
                .success_message = "Successfully added address ::1 to loopback interface",
167
                .eexist_message = "::1 has already been added to loopback interface",
168
        }, state_up = {
140✔
169
                .error_message = "Failed to bring loopback interface up",
170
                .success_message = "Successfully brought loopback interface up",
171
        };
172
        int r;
140✔
173

174
        /* Note, we, generally assume callers ignore the return code here (except test cases), hence only log add LOG_WARN level. */
175

176
        r = sd_netlink_open(&rtnl);
140✔
177
        if (r < 0)
140✔
UNCOV
178
                return log_warning_errno(r, "Failed to open netlink, ignoring: %m");
×
179

180
        /* Note that we add the IP addresses here explicitly even though the kernel does that too implicitly when
181
         * setting up the loopback device. The reason we do this here a second time (and possibly race against the
182
         * kernel) is that we want to synchronously wait until the IP addresses are set up correctly, see
183
         *
184
         * https://github.com/systemd/systemd/issues/5641 */
185

186
        r = add_ipv4_address(rtnl, &state_4);
140✔
187
        if (r < 0)
140✔
UNCOV
188
                return log_warning_errno(r, "Failed to enqueue IPv4 loopback address add request, ignoring: %m");
×
189

190
        r = add_ipv6_address(rtnl, &state_6);
140✔
191
        if (r < 0)
140✔
UNCOV
192
                return log_warning_errno(r, "Failed to enqueue IPv6 loopback address add request, ignoring: %m");
×
193

194
        r = start_loopback(rtnl, &state_up);
140✔
195
        if (r < 0)
140✔
UNCOV
196
                return log_warning_errno(r, "Failed to enqueue loopback interface start request, ignoring: %m");
×
197

198
        while (state_4.n_messages + state_6.n_messages + state_up.n_messages > 0) {
560✔
199
                r = sd_netlink_wait(rtnl, LOOPBACK_SETUP_TIMEOUT_USEC);
420✔
200
                if (r < 0)
420✔
201
                        return log_warning_errno(r, "Failed to wait for netlink event, ignoring: %m");
×
202

203
                r = sd_netlink_process(rtnl, NULL);
420✔
204
                if (r < 0)
420✔
UNCOV
205
                        return log_warning_errno(r, "Failed to process netlink event, ignoring: %m");
×
206
        }
207

208
        /* Note that we don't really care whether the addresses could be added or not */
209
        if (state_up.rcode != 0) {
140✔
210

211
                /* If we lack the permissions to configure the loopback device, but we find it to be already
212
                 * configured, let's exit cleanly, in order to supported unprivileged containers. */
UNCOV
213
                if (ERRNO_IS_PRIVILEGE(state_up.rcode)) {
×
UNCOV
214
                        r = check_loopback(rtnl);
×
UNCOV
215
                        if (r < 0)
×
UNCOV
216
                                log_debug_errno(r, "Failed to check if loopback device might already be up, ignoring: %m");
×
UNCOV
217
                        else if (r > 0) {
×
218
                                log_debug("Configuring loopback failed, but device is already up, suppressing failure.");
×
219
                                return 0;
×
220
                        }
221
                }
222

223
                return log_warning_errno(state_up.rcode, "Failed to configure loopback network device, ignoring: %m");
×
224
        }
225

226
        return 0;
227
}
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc