]> git.ipfire.org Git - thirdparty/systemd.git/blame - src/shared/loop-util.c
man/systemd-sysext: list ephemeral/ephemeral-import in the list of options
[thirdparty/systemd.git] / src / shared / loop-util.c
CommitLineData
db9ecf05 1/* SPDX-License-Identifier: LGPL-2.1-or-later */
8c1be37e 2
10c1b188
LP
3#if HAVE_VALGRIND_MEMCHECK_H
4#include <valgrind/memcheck.h>
5#endif
6
8c1be37e
LP
7#include <fcntl.h>
8#include <linux/loop.h>
441ec804 9#include <sys/file.h>
8c1be37e 10#include <sys/ioctl.h>
f2d9213f 11#include <unistd.h>
8c1be37e 12
021bf175
LP
13#include "sd-device.h"
14
8c1be37e 15#include "alloc-util.h"
86c1c1f3 16#include "blockdev-util.h"
fcd8a19d 17#include "data-fd-util.h"
021bf175 18#include "device-util.h"
7176f06c 19#include "devnum-util.h"
22ee78a8 20#include "dissect-image.h"
e8c7c4d9 21#include "env-util.h"
b0a94268 22#include "errno-util.h"
8c1be37e 23#include "fd-util.h"
f1443709 24#include "fileio.h"
1cf40697 25#include "fs-util.h"
8c1be37e 26#include "loop-util.h"
f1443709 27#include "parse-util.h"
e77cab82 28#include "path-util.h"
b202ec20 29#include "random-util.h"
3cc44114 30#include "stat-util.h"
f1443709 31#include "stdio-util.h"
f2d9213f 32#include "string-util.h"
69a283c5 33#include "time-util.h"
8c1be37e 34
e8af3bfd 35static void cleanup_clear_loop_close(int *fd) {
86c1c1f3
LP
36 if (*fd < 0)
37 return;
38
39 (void) ioctl(*fd, LOOP_CLR_FD);
40 (void) safe_close(*fd);
41}
42
021bf175
LP
43static int loop_is_bound(int fd) {
44 struct loop_info64 info;
45
8e398254 46 if (ioctl(ASSERT_FD(fd), LOOP_GET_STATUS64, &info) < 0) {
021bf175
LP
47 if (errno == ENXIO)
48 return false; /* not bound! */
49
50 return -errno;
51 }
52
53 return true; /* bound! */
54}
55
7f52206a 56static int open_lock_fd(int primary_fd, int operation) {
254d1313 57 _cleanup_close_ int lock_fd = -EBADF;
7f52206a 58
10719a6f 59 assert(IN_SET(operation & ~LOCK_NB, LOCK_SH, LOCK_EX));
7f52206a 60
8e398254 61 lock_fd = fd_reopen(ASSERT_FD(primary_fd), O_RDONLY|O_CLOEXEC|O_NONBLOCK|O_NOCTTY);
7f52206a
LP
62 if (lock_fd < 0)
63 return lock_fd;
10719a6f 64
7f52206a
LP
65 if (flock(lock_fd, operation) < 0)
66 return -errno;
67
10719a6f 68 return TAKE_FD(lock_fd);
7f52206a
LP
69}
70
54ba7daf 71static int loop_configure_verify_direct_io(int fd, const struct loop_config *c) {
ac110243 72 assert(fd >= 0);
54ba7daf
YW
73 assert(c);
74
75 if (FLAGS_SET(c->info.lo_flags, LO_FLAGS_DIRECT_IO)) {
76 struct loop_info64 info;
77
78 if (ioctl(fd, LOOP_GET_STATUS64, &info) < 0)
79 return log_debug_errno(errno, "Failed to issue LOOP_GET_STATUS64: %m");
80
81#if HAVE_VALGRIND_MEMCHECK_H
82 VALGRIND_MAKE_MEM_DEFINED(&info, sizeof(info));
83#endif
84
85 /* On older kernels (<= 5.3) it was necessary to set the block size of the loopback block
86 * device to the logical block size of the underlying file system. Since there was no nice
87 * way to query the value, we are not bothering to do this however. On newer kernels the
88 * block size is propagated automatically and does not require intervention from us. We'll
89 * check here if enabling direct IO worked, to make this easily debuggable however.
90 *
91 * (Should anyone really care and actually wants direct IO on old kernels: it might be worth
f5bb0a31
LB
92 * enabling direct IO with iteratively larger block sizes until it eventually works.)
93 *
94 * On older kernels (e.g.: 5.10) when this is attempted on a file stored on a dm-crypt
95 * backed partition the kernel will start returning I/O errors when accessing the mounted
96 * loop device, so return a recognizable error that causes the operation to be started
97 * from scratch without the LO_FLAGS_DIRECT_IO flag. */
54ba7daf 98 if (!FLAGS_SET(info.lo_flags, LO_FLAGS_DIRECT_IO))
f5bb0a31
LB
99 return log_debug_errno(
100 SYNTHETIC_ERRNO(ENOANO),
101 "Could not enable direct IO mode, retrying in buffered IO mode.");
54ba7daf
YW
102 }
103
104 return 0;
105}
106
107static int loop_configure_verify(int fd, const struct loop_config *c) {
108 bool broken = false;
109 int r;
110
111 assert(fd >= 0);
112 assert(c);
113
fd83c98e 114 if (c->block_size != 0) {
65046b92 115 uint32_t ssz;
fd83c98e 116
65046b92
LP
117 r = blockdev_get_sector_size(fd, &ssz);
118 if (r < 0)
119 return r;
fd83c98e 120
1163ddb3 121 if (ssz != c->block_size) {
65046b92 122 log_debug("LOOP_CONFIGURE didn't honour requested block size %" PRIu32 ", got %" PRIu32 " instead. Ignoring.", c->block_size, ssz);
1163ddb3
LP
123 broken = true;
124 }
fd83c98e
AD
125 }
126
54ba7daf
YW
127 if (c->info.lo_sizelimit != 0) {
128 /* Kernel 5.8 vanilla doesn't properly propagate the size limit into the
129 * block device. If it's used, let's immediately check if it had the desired
130 * effect hence. And if not use classic LOOP_SET_STATUS64. */
131 uint64_t z;
132
c961a8c6
LP
133 r = blockdev_get_device_size(fd, &z);
134 if (r < 0)
135 return r;
54ba7daf
YW
136
137 if (z != c->info.lo_sizelimit) {
fd83c98e 138 log_debug("LOOP_CONFIGURE is broken, doesn't honour .info.lo_sizelimit. Falling back to LOOP_SET_STATUS64.");
54ba7daf
YW
139 broken = true;
140 }
141 }
142
143 if (FLAGS_SET(c->info.lo_flags, LO_FLAGS_PARTSCAN)) {
144 /* Kernel 5.8 vanilla doesn't properly propagate the partition scanning flag
145 * into the block device. Let's hence verify if things work correctly here
146 * before returning. */
147
bff5d2fd 148 r = blockdev_partscan_enabled_fd(fd);
54ba7daf
YW
149 if (r < 0)
150 return r;
151 if (r == 0) {
152 log_debug("LOOP_CONFIGURE is broken, doesn't honour LO_FLAGS_PARTSCAN. Falling back to LOOP_SET_STATUS64.");
153 broken = true;
154 }
155 }
156
157 r = loop_configure_verify_direct_io(fd, c);
158 if (r < 0)
159 return r;
160
161 return !broken;
162}
163
164static int loop_configure_fallback(int fd, const struct loop_config *c) {
165 struct loop_info64 info_copy;
1163ddb3 166 int r;
54ba7daf
YW
167
168 assert(fd >= 0);
169 assert(c);
170
171 /* Only some of the flags LOOP_CONFIGURE can set are also settable via LOOP_SET_STATUS64, hence mask
172 * them out. */
173 info_copy = c->info;
174 info_copy.lo_flags &= LOOP_SET_STATUS_SETTABLE_FLAGS;
175
176 /* Since kernel commit 5db470e229e22b7eda6e23b5566e532c96fb5bc3 (kernel v5.0) the LOOP_SET_STATUS64
fd83c98e 177 * ioctl can return EAGAIN in case we change the info.lo_offset field, if someone else is accessing the
54ba7daf
YW
178 * block device while we try to reconfigure it. This is a pretty common case, since udev might
179 * instantly start probing the device as soon as we attach an fd to it. Hence handle it in two ways:
180 * first, let's take the BSD lock to ensure that udev will not step in between the point in
181 * time where we attach the fd and where we reconfigure the device. Secondly, let's wait 50ms on
182 * EAGAIN and retry. The former should be an efficient mechanism to avoid we have to wait 50ms
183 * needlessly if we are just racing against udev. The latter is protection against all other cases,
184 * i.e. peers that do not take the BSD lock. */
185
186 for (unsigned n_attempts = 0;;) {
187 if (ioctl(fd, LOOP_SET_STATUS64, &info_copy) >= 0)
188 break;
189
190 if (errno != EAGAIN || ++n_attempts >= 64)
191 return log_debug_errno(errno, "Failed to configure loopback block device: %m");
192
193 /* Sleep some random time, but at least 10ms, at most 250ms. Increase the delay the more
194 * failed attempts we see */
4251512e 195 (void) usleep_safe(UINT64_C(10) * USEC_PER_MSEC +
54ba7daf
YW
196 random_u64_range(UINT64_C(240) * USEC_PER_MSEC * n_attempts/64));
197 }
198
1163ddb3
LP
199 /* If a block size is requested then try to configure it. If that doesn't work, ignore errors, but
200 * afterwards, let's validate what is in effect, and if it doesn't match what we want, fail */
201 if (c->block_size != 0) {
202 uint32_t ssz;
203
204 if (ioctl(fd, LOOP_SET_BLOCK_SIZE, (unsigned long) c->block_size) < 0)
205 log_debug_errno(errno, "Failed to set sector size, ignoring: %m");
206
207 r = blockdev_get_sector_size(fd, &ssz);
208 if (r < 0)
209 return log_debug_errno(r, "Failed to read sector size: %m");
210 if (ssz != c->block_size)
211 return log_debug_errno(SYNTHETIC_ERRNO(EIO), "Sector size of loopback device doesn't match what we requested, refusing.");
212 }
213
54ba7daf
YW
214 /* LO_FLAGS_DIRECT_IO is a flags we need to configure via explicit ioctls. */
215 if (FLAGS_SET(c->info.lo_flags, LO_FLAGS_DIRECT_IO))
216 if (ioctl(fd, LOOP_SET_DIRECT_IO, 1UL) < 0)
217 log_debug_errno(errno, "Failed to enable direct IO mode, ignoring: %m");
218
219 return loop_configure_verify_direct_io(fd, c);
220}
221
95c50092 222static int loop_configure(
021bf175 223 int nr,
da4fd288
YW
224 int open_flags,
225 int lock_op,
95c50092 226 const struct loop_config *c,
da4fd288 227 LoopDevice **ret) {
95c50092 228
bb273a51
YW
229 static bool loop_configure_broken = false;
230
da4fd288 231 _cleanup_(sd_device_unrefp) sd_device *dev = NULL;
254d1313
ZJS
232 _cleanup_(cleanup_clear_loop_close) int loop_with_fd = -EBADF; /* This must be declared before lock_fd. */
233 _cleanup_close_ int fd = -EBADF, lock_fd = -EBADF;
da4fd288 234 _cleanup_free_ char *node = NULL;
7ba1816d 235 uint64_t diskseq = 0;
da4fd288 236 dev_t devno;
86c1c1f3
LP
237 int r;
238
021bf175 239 assert(nr >= 0);
86c1c1f3 240 assert(c);
da4fd288
YW
241 assert(ret);
242
243 if (asprintf(&node, "/dev/loop%i", nr) < 0)
432f1fa8 244 return log_oom_debug();
da4fd288
YW
245
246 r = sd_device_new_from_devname(&dev, node);
247 if (r < 0)
432f1fa8 248 return log_debug_errno(r, "Failed to create sd_device object for \"%s\": %m", node);
da4fd288
YW
249
250 r = sd_device_get_devnum(dev, &devno);
251 if (r < 0)
432f1fa8 252 return log_device_debug_errno(dev, r, "Failed to get devnum: %m");
da4fd288
YW
253
254 fd = sd_device_open(dev, O_CLOEXEC|O_NONBLOCK|O_NOCTTY|open_flags);
255 if (fd < 0)
432f1fa8 256 return log_device_debug_errno(dev, fd, "Failed to open device: %m");
95c50092 257
021bf175
LP
258 /* Let's lock the device before we do anything. We take the BSD lock on a second, separately opened
259 * fd for the device. udev after all watches for close() events (specifically IN_CLOSE_WRITE) on
260 * block devices to reprobe them, hence by having a separate fd we will later close() we can ensure
261 * we trigger udev after everything is done. If we'd lock our own fd instead and keep it open for a
262 * long time udev would possibly never run on it again, even though the fd is unlocked, simply
263 * because we never close() it. It also has the nice benefit we can use the _cleanup_close_ logic to
264 * automatically release the lock, after we are done. */
7f52206a 265 lock_fd = open_lock_fd(fd, LOCK_EX);
021bf175 266 if (lock_fd < 0)
432f1fa8
YW
267 return log_device_debug_errno(dev, lock_fd, "Failed to acquire lock: %m");
268
269 log_device_debug(dev, "Acquired exclusive lock.");
021bf175 270
53274fdb
YW
271 /* Let's see if backing file is really unattached. Someone may already attach a backing file without
272 * taking BSD lock. */
273 r = loop_is_bound(fd);
274 if (r < 0)
432f1fa8 275 return log_device_debug_errno(dev, r, "Failed to check if the loopback block device is bound: %m");
53274fdb 276 if (r > 0)
432f1fa8
YW
277 return log_device_debug_errno(dev, SYNTHETIC_ERRNO(EBUSY),
278 "The loopback block device is already bound, ignoring.");
53274fdb 279
021bf175
LP
280 /* Let's see if the device is really detached, i.e. currently has no associated partition block
281 * devices. On various kernels (such as 5.8) it is possible to have a loopback block device that
247738b4
LP
282 * superficially is detached but still has partition block devices associated for it. Let's then
283 * manually remove the partitions via BLKPG, and tell the caller we did that via EUCLEAN, so they try
284 * again. */
833106b8 285 r = block_device_remove_all_partitions(dev, fd);
021bf175 286 if (r < 0)
432f1fa8 287 return log_device_debug_errno(dev, r, "Failed to remove partitions on the loopback block device: %m");
833106b8
YW
288 if (r > 0)
289 /* Removed all partitions. Let's report this to the caller, to try again, and count this as
53274fdb 290 * an attempt. */
432f1fa8
YW
291 return log_device_debug_errno(dev, SYNTHETIC_ERRNO(EUCLEAN),
292 "Removed partitions on the loopback block device.");
021bf175 293
bb273a51 294 if (!loop_configure_broken) {
95c50092 295 if (ioctl(fd, LOOP_CONFIGURE, c) < 0) {
9a7b20b6
MY
296 /* Do fallback only if LOOP_CONFIGURE is not supported, propagate all other errors. */
297 if (!ERRNO_IS_IOCTL_NOT_SUPPORTED(errno))
432f1fa8 298 return log_device_debug_errno(dev, errno, "ioctl(LOOP_CONFIGURE) failed: %m");
86c1c1f3 299
bb273a51 300 loop_configure_broken = true;
95c50092 301 } else {
da4fd288
YW
302 loop_with_fd = TAKE_FD(fd);
303
304 r = loop_configure_verify(loop_with_fd, c);
54ba7daf 305 if (r < 0)
432f1fa8 306 return log_device_debug_errno(dev, r, "Failed to verify if loopback block device is correctly configured: %m");
54ba7daf 307 if (r == 0) {
95c50092 308 /* LOOP_CONFIGURE doesn't work. Remember that. */
bb273a51 309 loop_configure_broken = true;
95c50092
LP
310
311 /* We return EBUSY here instead of retrying immediately with LOOP_SET_FD,
312 * because LOOP_CLR_FD is async: if the operation cannot be executed right
313 * away it just sets the autoclear flag on the device. This means there's a
314 * good chance we cannot actually reuse the loopback device right-away. Hence
315 * let's assume it's busy, avoid the trouble and let the calling loop call us
316 * again with a new, likely unused device. */
da4fd288 317 return -EBUSY;
bb2551bd 318 }
95c50092 319 }
86c1c1f3
LP
320 }
321
bb273a51 322 if (loop_configure_broken) {
bb273a51 323 if (ioctl(fd, LOOP_SET_FD, c->fd) < 0)
432f1fa8 324 return log_device_debug_errno(dev, errno, "ioctl(LOOP_SET_FD) failed: %m");
86c1c1f3 325
da4fd288
YW
326 loop_with_fd = TAKE_FD(fd);
327
328 r = loop_configure_fallback(loop_with_fd, c);
bb273a51 329 if (r < 0)
da4fd288 330 return r;
bb273a51 331 }
e8c7c4d9 332
da4fd288
YW
333 r = fd_get_diskseq(loop_with_fd, &diskseq);
334 if (r < 0 && r != -EOPNOTSUPP)
432f1fa8 335 return log_device_debug_errno(dev, r, "Failed to get diskseq: %m");
31c75fcc 336
da4fd288
YW
337 switch (lock_op & ~LOCK_NB) {
338 case LOCK_EX: /* Already in effect */
339 break;
340 case LOCK_SH: /* Downgrade */
341 if (flock(lock_fd, lock_op) < 0)
432f1fa8 342 return log_device_debug_errno(dev, errno, "Failed to downgrade lock level: %m");
da4fd288
YW
343 break;
344 case LOCK_UN: /* Release */
345 lock_fd = safe_close(lock_fd);
346 break;
347 default:
348 assert_not_reached();
349 }
350
c961a8c6
LP
351 uint64_t device_size;
352 r = blockdev_get_device_size(loop_with_fd, &device_size);
353 if (r < 0)
354 return log_device_debug_errno(dev, r, "Failed to get loopback device size: %m");
355
da4fd288
YW
356 LoopDevice *d = new(LoopDevice, 1);
357 if (!d)
432f1fa8 358 return log_oom_debug();
da4fd288
YW
359
360 *d = (LoopDevice) {
36d5eb0b 361 .n_ref = 1,
da4fd288
YW
362 .fd = TAKE_FD(loop_with_fd),
363 .lock_fd = TAKE_FD(lock_fd),
364 .node = TAKE_PTR(node),
365 .nr = nr,
366 .devno = devno,
367 .dev = TAKE_PTR(dev),
368 .diskseq = diskseq,
22ee78a8 369 .sector_size = c->block_size,
c961a8c6 370 .device_size = device_size,
6bc20134 371 .created = true,
da4fd288 372 };
86c1c1f3 373
da4fd288
YW
374 *ret = TAKE_PTR(d);
375 return 0;
e8af3bfd
ZJS
376}
377
e8c7c4d9 378static int loop_device_make_internal(
e77cab82 379 const char *path,
ed9eeb7b
LP
380 int fd,
381 int open_flags,
382 uint64_t offset,
383 uint64_t size,
22ee78a8 384 uint32_t sector_size,
ed9eeb7b 385 uint32_t loop_flags,
7f52206a 386 int lock_op,
ed9eeb7b 387 LoopDevice **ret) {
8c1be37e 388
da4fd288 389 _cleanup_(loop_device_unrefp) LoopDevice *d = NULL;
f5bb0a31 390 _cleanup_close_ int reopened_fd = -EBADF, control = -EBADF;
da4fd288 391 _cleanup_free_ char *backing_file = NULL;
86c1c1f3 392 struct loop_config config;
da4fd288 393 int r, f_flags;
8c1be37e 394 struct stat st;
8c1be37e 395
8c1be37e
LP
396 assert(ret);
397 assert(IN_SET(open_flags, O_RDWR, O_RDONLY));
398
8e398254 399 if (fstat(ASSERT_FD(fd), &st) < 0)
8c1be37e
LP
400 return -errno;
401
402 if (S_ISBLK(st.st_mode)) {
1996ad28 403 if (offset == 0 && IN_SET(size, 0, UINT64_MAX))
d7654742
LP
404 /* If this is already a block device and we are supposed to cover the whole of it
405 * then store an fd to the original open device node — and do not actually create an
1996ad28 406 * unnecessary loopback device for it. */
de3b7f16 407 return loop_device_open_from_fd(fd, open_flags, lock_op, ret);
ed9eeb7b
LP
408 } else {
409 r = stat_verify_regular(&st);
410 if (r < 0)
411 return r;
8c1be37e
LP
412 }
413
e77cab82
YW
414 if (path) {
415 r = path_make_absolute_cwd(path, &backing_file);
416 if (r < 0)
417 return r;
418
419 path_simplify(backing_file);
420 } else {
421 r = fd_get_path(fd, &backing_file);
422 if (r < 0)
423 return r;
424 }
425
e8c7c4d9
LP
426 f_flags = fcntl(fd, F_GETFL);
427 if (f_flags < 0)
428 return -errno;
429
430 if (FLAGS_SET(loop_flags, LO_FLAGS_DIRECT_IO) != FLAGS_SET(f_flags, O_DIRECT)) {
431 /* If LO_FLAGS_DIRECT_IO is requested, then make sure we have the fd open with O_DIRECT, as
432 * that's required. Conversely, if it's off require that O_DIRECT is off too (that's because
433 * new kernels will implicitly enable LO_FLAGS_DIRECT_IO if O_DIRECT is set).
434 *
435 * Our intention here is that LO_FLAGS_DIRECT_IO is the primary knob, and O_DIRECT derived
436 * from that automatically. */
437
f5bb0a31
LB
438 reopened_fd = fd_reopen(fd, (FLAGS_SET(loop_flags, LO_FLAGS_DIRECT_IO) ? O_DIRECT : 0)|O_CLOEXEC|O_NONBLOCK|open_flags);
439 if (reopened_fd < 0) {
e8c7c4d9 440 if (!FLAGS_SET(loop_flags, LO_FLAGS_DIRECT_IO))
d579c42e 441 return log_debug_errno(reopened_fd, "Failed to reopen file descriptor without O_DIRECT: %m");
e8c7c4d9
LP
442
443 /* Some file systems might not support O_DIRECT, let's gracefully continue without it then. */
d579c42e 444 log_debug_errno(reopened_fd, "Failed to enable O_DIRECT for backing file descriptor for loopback device. Continuing without.");
e8c7c4d9
LP
445 loop_flags &= ~LO_FLAGS_DIRECT_IO;
446 } else
f5bb0a31 447 fd = reopened_fd; /* From now on, operate on our new O_DIRECT fd */
e8c7c4d9
LP
448 }
449
8c1be37e
LP
450 control = open("/dev/loop-control", O_RDWR|O_CLOEXEC|O_NOCTTY|O_NONBLOCK);
451 if (control < 0)
452 return -errno;
453
22ee78a8
LP
454 if (sector_size == 0)
455 /* If no sector size is specified, default to the classic default */
456 sector_size = 512;
457 else if (sector_size == UINT32_MAX) {
458
459 if (S_ISBLK(st.st_mode))
460 /* If the sector size is specified as UINT32_MAX we'll propagate the sector size of
461 * the underlying block device. */
462 r = blockdev_get_sector_size(fd, &sector_size);
463 else {
92651a7a 464 _cleanup_close_ int non_direct_io_fd = -EBADF;
22ee78a8
LP
465 int probe_fd;
466
467 assert(S_ISREG(st.st_mode));
468
469 /* If sector size is specified as UINT32_MAX, we'll try to probe the right sector
470 * size of the image in question by looking for the GPT partition header at various
471 * offsets. This of course only works if the image already has a disk label.
472 *
473 * So here we actually want to read the file contents ourselves. This is quite likely
474 * not going to work if we managed to enable O_DIRECT, because in such a case there
475 * are some pretty strict alignment requirements to offset, size and target, but
476 * there's no way to query what alignment specifically is actually required. Hence,
477 * let's avoid the mess, and temporarily open an fd without O_DIRECT for the probing
478 * logic. */
479
480 if (FLAGS_SET(loop_flags, LO_FLAGS_DIRECT_IO)) {
481 non_direct_io_fd = fd_reopen(fd, O_RDONLY|O_CLOEXEC|O_NONBLOCK);
482 if (non_direct_io_fd < 0)
483 return non_direct_io_fd;
484
485 probe_fd = non_direct_io_fd;
486 } else
487 probe_fd = fd;
488
489 r = probe_sector_size(probe_fd, &sector_size);
490 }
491 if (r < 0)
492 return r;
493 }
494
86c1c1f3
LP
495 config = (struct loop_config) {
496 .fd = fd,
22ee78a8 497 .block_size = sector_size,
86c1c1f3
LP
498 .info = {
499 /* Use the specified flags, but configure the read-only flag from the open flags, and force autoclear */
b1236ce3 500 .lo_flags = (loop_flags & ~LO_FLAGS_READ_ONLY) | ((open_flags & O_ACCMODE_STRICT) == O_RDONLY ? LO_FLAGS_READ_ONLY : 0) | LO_FLAGS_AUTOCLEAR,
86c1c1f3
LP
501 .lo_offset = offset,
502 .lo_sizelimit = size == UINT64_MAX ? 0 : size,
503 },
504 };
505
0f6519d4
LP
506 /* Loop around LOOP_CTL_GET_FREE, since at the moment we attempt to open the returned device it might
507 * be gone already, taken by somebody else racing against us. */
e8af3bfd 508 for (unsigned n_attempts = 0;;) {
432f1fa8 509 usec_t usec;
da4fd288 510 int nr;
e8af3bfd 511
cc530466
LP
512 /* Let's take a lock on the control device first. On a busy system, where many programs
513 * attempt to allocate a loopback device at the same time, we might otherwise keep looping
514 * around relatively heavy operations: asking for a free loopback device, then opening it,
515 * validating it, attaching something to it. Let's serialize this whole operation, to make
516 * unnecessary busywork less likely. Note that this is just something we do to optimize our
517 * own code (and whoever else decides to use LOCK_EX locks for this), taking this lock is not
518 * necessary, it just means it's less likely we have to iterate through this loop again and
4c1d50e6
LP
519 * again if our own code races against our own code.
520 *
521 * Note: our lock protocol is to take the /dev/loop-control lock first, and the block device
522 * lock second, if both are taken, and always in this order, to avoid ABBA locking issues. */
cc530466
LP
523 if (flock(control, LOCK_EX) < 0)
524 return -errno;
525
0f6519d4
LP
526 nr = ioctl(control, LOOP_CTL_GET_FREE);
527 if (nr < 0)
528 return -errno;
8c1be37e 529
da4fd288
YW
530 r = loop_configure(nr, open_flags, lock_op, &config, &d);
531 if (r >= 0)
532 break;
cc5bae6c 533
da4fd288
YW
534 /* -ENODEV or friends: Somebody might've gotten the same number from the kernel, used the
535 * device, and called LOOP_CTL_REMOVE on it. Let's retry with a new number.
536 * -EBUSY: a file descriptor is already bound to the loopback block device.
f5bb0a31
LB
537 * -EUCLEAN: some left-over partition devices that were cleaned up.
538 * -ENOANO: we tried to use LO_FLAGS_DIRECT_IO but the kernel rejected it. */
539 if (!ERRNO_IS_DEVICE_ABSENT(r) && !IN_SET(r, -EBUSY, -EUCLEAN, -ENOANO))
2421dd72 540 return r;
01813148 541
cc530466
LP
542 /* OK, this didn't work, let's try again a bit later, but first release the lock on the
543 * control device */
544 if (flock(control, LOCK_UN) < 0)
545 return -errno;
546
e8af3bfd
ZJS
547 if (++n_attempts >= 64) /* Give up eventually */
548 return -EBUSY;
0f6519d4 549
f5bb0a31
LB
550 /* If we failed to enable direct IO mode, let's retry without it. We restart the process as
551 * on some combination of kernel version and storage filesystem, the kernel is very unhappy
552 * about a failed DIRECT_IO enablement and throws I/O errors. */
553 if (r == -ENOANO && FLAGS_SET(config.info.lo_flags, LO_FLAGS_DIRECT_IO)) {
554 config.info.lo_flags &= ~LO_FLAGS_DIRECT_IO;
555 open_flags &= ~O_DIRECT;
556
557 int non_direct_io_fd = fd_reopen(config.fd, O_CLOEXEC|O_NONBLOCK|open_flags);
558 if (non_direct_io_fd < 0)
559 return log_debug_errno(
560 non_direct_io_fd,
561 "Failed to reopen file descriptor without O_DIRECT: %m");
562
563 safe_close(reopened_fd);
564 fd = config.fd = /* For cleanups */ reopened_fd = non_direct_io_fd;
565 }
566
b202ec20
LP
567 /* Wait some random time, to make collision less likely. Let's pick a random time in the
568 * range 0ms…250ms, linearly scaled by the number of failed attempts. */
432f1fa8
YW
569 usec = random_u64_range(UINT64_C(10) * USEC_PER_MSEC +
570 UINT64_C(240) * USEC_PER_MSEC * n_attempts/64);
571 log_debug("Trying again after %s.", FORMAT_TIMESPAN(usec, USEC_PER_MSEC));
4251512e 572 (void) usleep_safe(usec);
0f6519d4 573 }
8c1be37e 574
da4fd288 575 d->backing_file = TAKE_PTR(backing_file);
4d2a9e3e
LP
576 d->backing_inode = st.st_ino;
577 d->backing_devno = st.st_dev;
8c1be37e 578
3b195f63
LP
579 log_debug("Successfully acquired %s, devno=%u:%u, nr=%i, diskseq=%" PRIu64,
580 d->node,
581 major(d->devno), minor(d->devno),
582 d->nr,
583 d->diskseq);
584
da4fd288
YW
585 *ret = TAKE_PTR(d);
586 return 0;
8c1be37e
LP
587}
588
e8c7c4d9
LP
589static uint32_t loop_flags_mangle(uint32_t loop_flags) {
590 int r;
591
592 r = getenv_bool("SYSTEMD_LOOP_DIRECT_IO");
593 if (r < 0 && r != -ENXIO)
594 log_debug_errno(r, "Failed to parse $SYSTEMD_LOOP_DIRECT_IO, ignoring: %m");
595
bfd08445 596 return UPDATE_FLAG(loop_flags, LO_FLAGS_DIRECT_IO, r != 0); /* Turn on LO_FLAGS_DIRECT_IO by default, unless explicitly configured to off. */
e8c7c4d9
LP
597}
598
599int loop_device_make(
600 int fd,
601 int open_flags,
602 uint64_t offset,
603 uint64_t size,
22ee78a8 604 uint32_t sector_size,
e8c7c4d9 605 uint32_t loop_flags,
7f52206a 606 int lock_op,
e8c7c4d9
LP
607 LoopDevice **ret) {
608
609 assert(fd >= 0);
610 assert(ret);
e8c7c4d9
LP
611
612 return loop_device_make_internal(
e77cab82 613 NULL,
e8c7c4d9
LP
614 fd,
615 open_flags,
616 offset,
617 size,
22ee78a8 618 sector_size,
bfd08445 619 loop_flags_mangle(loop_flags),
7f52206a 620 lock_op,
e8c7c4d9
LP
621 ret);
622}
623
972c8db5
DDM
624int loop_device_make_by_path_at(
625 int dir_fd,
79e8393a
LP
626 const char *path,
627 int open_flags,
22ee78a8 628 uint32_t sector_size,
79e8393a 629 uint32_t loop_flags,
7f52206a 630 int lock_op,
79e8393a
LP
631 LoopDevice **ret) {
632
e8c7c4d9 633 int r, basic_flags, direct_flags, rdwr_flags;
254d1313 634 _cleanup_close_ int fd = -EBADF;
aa4d3aa3 635 bool direct = false;
8c1be37e 636
972c8db5 637 assert(dir_fd >= 0 || dir_fd == AT_FDCWD);
8c1be37e
LP
638 assert(path);
639 assert(ret);
b0a94268 640 assert(open_flags < 0 || IN_SET(open_flags, O_RDWR, O_RDONLY));
8c1be37e 641
b0a94268
LP
642 /* Passing < 0 as open_flags here means we'll try to open the device writable if we can, retrying
643 * read-only if we cannot. */
644
e8c7c4d9
LP
645 loop_flags = loop_flags_mangle(loop_flags);
646
647 /* Let's open with O_DIRECT if we can. But not all file systems support that, hence fall back to
648 * non-O_DIRECT mode automatically, if it fails. */
649
650 basic_flags = O_CLOEXEC|O_NONBLOCK|O_NOCTTY;
651 direct_flags = FLAGS_SET(loop_flags, LO_FLAGS_DIRECT_IO) ? O_DIRECT : 0;
652 rdwr_flags = open_flags >= 0 ? open_flags : O_RDWR;
653
e40b11be 654 fd = xopenat(dir_fd, path, basic_flags|direct_flags|rdwr_flags);
e8c7c4d9 655 if (fd < 0 && direct_flags != 0) /* If we had O_DIRECT on, and things failed with that, let's immediately try again without */
e40b11be 656 fd = xopenat(dir_fd, path, basic_flags|rdwr_flags);
aa4d3aa3
LP
657 else
658 direct = direct_flags != 0;
b0a94268 659 if (fd < 0) {
6383abd6 660 r = fd;
b0a94268
LP
661
662 /* Retry read-only? */
663 if (open_flags >= 0 || !(ERRNO_IS_PRIVILEGE(r) || r == -EROFS))
664 return r;
665
e40b11be 666 fd = xopenat(dir_fd, path, basic_flags|direct_flags|O_RDONLY);
e8c7c4d9 667 if (fd < 0 && direct_flags != 0) /* as above */
e40b11be 668 fd = xopenat(dir_fd, path, basic_flags|O_RDONLY);
aa4d3aa3
LP
669 else
670 direct = direct_flags != 0;
b0a94268
LP
671 if (fd < 0)
672 return r; /* Propagate original error */
673
674 open_flags = O_RDONLY;
675 } else if (open_flags < 0)
676 open_flags = O_RDWR;
8c1be37e 677
aa4d3aa3
LP
678 log_debug("Opened '%s' in %s access mode%s, with O_DIRECT %s%s.",
679 path,
680 open_flags == O_RDWR ? "O_RDWR" : "O_RDONLY",
681 open_flags != rdwr_flags ? " (O_RDWR was requested but not allowed)" : "",
682 direct ? "enabled" : "disabled",
683 direct != (direct_flags != 0) ? " (O_DIRECT was requested but not supported)" : "");
684
972c8db5
DDM
685 return loop_device_make_internal(
686 dir_fd == AT_FDCWD ? path : NULL,
687 fd,
688 open_flags,
689 /* offset = */ 0,
690 /* size = */ 0,
691 sector_size,
692 loop_flags,
693 lock_op,
694 ret);
8c1be37e
LP
695}
696
fcd8a19d
LP
697int loop_device_make_by_path_memory(
698 const char *path,
699 int open_flags,
22ee78a8 700 uint32_t sector_size,
fcd8a19d
LP
701 uint32_t loop_flags,
702 int lock_op,
703 LoopDevice **ret) {
704
705 _cleanup_close_ int fd = -EBADF, mfd = -EBADF;
706 _cleanup_free_ char *fn = NULL;
707 struct stat st;
708 int r;
709
710 assert(path);
711 assert(IN_SET(open_flags, O_RDWR, O_RDONLY));
712 assert(ret);
713
714 loop_flags &= ~LO_FLAGS_DIRECT_IO; /* memfds don't support O_DIRECT, hence LO_FLAGS_DIRECT_IO can't be used either */
715
716 fd = open(path, O_CLOEXEC|O_NONBLOCK|O_NOCTTY|O_RDONLY);
717 if (fd < 0)
718 return -errno;
719
720 if (fstat(fd, &st) < 0)
721 return -errno;
722
723 if (!S_ISREG(st.st_mode) && !S_ISBLK(st.st_mode))
724 return -EBADF;
725
726 r = path_extract_filename(path, &fn);
727 if (r < 0)
728 return r;
729
730 mfd = memfd_clone_fd(fd, fn, open_flags|O_CLOEXEC);
731 if (mfd < 0)
732 return mfd;
733
734 fd = safe_close(fd); /* Let's close the original early */
735
22ee78a8 736 return loop_device_make_internal(NULL, mfd, open_flags, 0, 0, sector_size, loop_flags, lock_op, ret);
fcd8a19d
LP
737}
738
36d5eb0b 739static LoopDevice* loop_device_free(LoopDevice *d) {
5bb1d7fb 740 _cleanup_close_ int control = -EBADF;
3a6ed1e1
LP
741 int r;
742
8c1be37e
LP
743 if (!d)
744 return NULL;
745
4c1d50e6
LP
746 /* Release any lock we might have on the device first. We want to open+lock the /dev/loop-control
747 * device below, but our lock protocol says that if both control and block device locks are taken,
748 * the control lock needs to be taken first, the block device lock second — in order to avoid ABBA
749 * locking issues. Moreover, we want to issue LOOP_CLR_FD on the block device further down, and that
750 * would fail if we had another fd open to the device. */
7f52206a
LP
751 d->lock_fd = safe_close(d->lock_fd);
752
4c1d50e6
LP
753 /* Let's open the control device early, and lock it, so that we can release our block device and
754 * delete it in a synchronized fashion, and allocators won't needlessly see the block device as free
755 * while we are about to delete it. */
7cb349f0 756 if (!LOOP_DEVICE_IS_FOREIGN(d) && !d->relinquished) {
4c1d50e6
LP
757 control = open("/dev/loop-control", O_RDWR|O_CLOEXEC|O_NOCTTY|O_NONBLOCK);
758 if (control < 0)
759 log_debug_errno(errno, "Failed to open loop control device, cannot remove loop device '%s', ignoring: %m", strna(d->node));
760 else if (flock(control, LOCK_EX) < 0)
761 log_debug_errno(errno, "Failed to lock loop control device, ignoring: %m");
762 }
763
764 /* Then let's release the loopback block device */
8c1be37e 765 if (d->fd >= 0) {
cae1e8fb
LP
766 /* Implicitly sync the device, since otherwise in-flight blocks might not get written */
767 if (fsync(d->fd) < 0)
768 log_debug_errno(errno, "Failed to sync loop block device, ignoring: %m");
769
7cb349f0 770 if (!LOOP_DEVICE_IS_FOREIGN(d) && !d->relinquished) {
3a6ed1e1
LP
771 /* We are supposed to clear the loopback device. Let's do this synchronously: lock
772 * the device, manually remove all partitions and then clear it. This should ensure
773 * udev doesn't concurrently access the devices, and we can be reasonably sure that
774 * once we are done here the device is cleared and all its partition children
775 * removed. Note that we lock our primary device fd here (and not a separate locking
776 * fd, as we do during allocation, since we want to keep the lock all the way through
777 * the LOOP_CLR_FD, but that call would fail if we had more than one fd open.) */
8c1be37e 778
3a6ed1e1
LP
779 if (flock(d->fd, LOCK_EX) < 0)
780 log_debug_errno(errno, "Failed to lock loop block device, ignoring: %m");
781
46c3a288 782 r = block_device_remove_all_partitions(d->dev, d->fd);
3a6ed1e1
LP
783 if (r < 0)
784 log_debug_errno(r, "Failed to remove partitions of loopback block device, ignoring: %m");
785
786 if (ioctl(d->fd, LOOP_CLR_FD) < 0)
787 log_debug_errno(errno, "Failed to clear loop device, ignoring: %m");
8c1be37e
LP
788 }
789
790 safe_close(d->fd);
791 }
792
4c1d50e6 793 /* Now that the block device is released, let's also try to remove it */
afbe20b7 794 if (control >= 0) {
6483bcef
ZJS
795 useconds_t delay = 5 * USEC_PER_MSEC; /* A total delay of 5090 ms between 39 attempts,
796 * (4*5 + 5*10 + 5*20 + … + 3*640) = 5090. */
afbe20b7
ZJS
797
798 for (unsigned attempt = 1;; attempt++) {
4c1d50e6
LP
799 if (ioctl(control, LOOP_CTL_REMOVE, d->nr) >= 0)
800 break;
afbe20b7 801 if (errno != EBUSY || attempt > 38) {
4c1d50e6
LP
802 log_debug_errno(errno, "Failed to remove device %s: %m", strna(d->node));
803 break;
f2d9213f 804 }
afbe20b7
ZJS
805 if (attempt % 5 == 0) {
806 log_debug("Device is still busy after %u attempts…", attempt);
807 delay *= 2;
808 }
809
4251512e 810 (void) usleep_safe(delay);
4c1d50e6 811 }
afbe20b7 812 }
8c1be37e
LP
813
814 free(d->node);
cc5bae6c 815 sd_device_unref(d->dev);
e77cab82 816 free(d->backing_file);
5fecf46d 817 return mfree(d);
8c1be37e 818}
a2ea3b2f 819
36d5eb0b
YW
820DEFINE_TRIVIAL_REF_UNREF_FUNC(LoopDevice, loop_device, loop_device_free);
821
a2ea3b2f
LP
822void loop_device_relinquish(LoopDevice *d) {
823 assert(d);
824
825 /* Don't attempt to clean up the loop device anymore from this point on. Leave the clean-ing up to the kernel
826 * itself, using the loop device "auto-clear" logic we already turned on when creating the device. */
827
828 d->relinquished = true;
829}
9dabc4fd 830
24d59aee
DDM
831void loop_device_unrelinquish(LoopDevice *d) {
832 assert(d);
833 d->relinquished = false;
834}
835
4f0ad43e
YW
836int loop_device_open(
837 sd_device *dev,
7f52206a
LP
838 int open_flags,
839 int lock_op,
840 LoopDevice **ret) {
841
254d1313 842 _cleanup_close_ int fd = -EBADF, lock_fd = -EBADF;
4f0ad43e 843 _cleanup_free_ char *node = NULL, *backing_file = NULL;
4d2a9e3e 844 dev_t devnum, backing_devno = 0;
b26c39ad 845 struct loop_info64 info;
4d2a9e3e 846 ino_t backing_inode = 0;
ffcb3324 847 uint64_t diskseq = 0;
9dabc4fd 848 LoopDevice *d;
4f0ad43e 849 const char *s;
a8d8a619 850 int r, nr = -1;
9dabc4fd 851
4f0ad43e 852 assert(dev);
e8c7c4d9 853 assert(IN_SET(open_flags, O_RDWR, O_RDONLY));
9dabc4fd
LP
854 assert(ret);
855
4f0ad43e
YW
856 /* Even if fd is provided through the argument in loop_device_open_from_fd(), we reopen the inode
857 * here, instead of keeping just a dup() clone of it around, since we want to ensure that the
858 * O_DIRECT flag of the handle we keep is off, we have our own file index, and have the right
859 * read/write mode in effect. */
860 fd = sd_device_open(dev, O_CLOEXEC|O_NONBLOCK|O_NOCTTY|open_flags);
861 if (fd < 0)
862 return fd;
cc5bae6c 863
4f0ad43e
YW
864 if ((lock_op & ~LOCK_NB) != LOCK_UN) {
865 lock_fd = open_lock_fd(fd, lock_op);
866 if (lock_fd < 0)
867 return lock_fd;
a8d8a619
YW
868 }
869
4f0ad43e 870 if (ioctl(fd, LOOP_GET_STATUS64, &info) >= 0) {
10c1b188
LP
871#if HAVE_VALGRIND_MEMCHECK_H
872 /* Valgrind currently doesn't know LOOP_GET_STATUS64. Remove this once it does */
873 VALGRIND_MAKE_MEM_DEFINED(&info, sizeof(info));
874#endif
b26c39ad 875 nr = info.lo_number;
e77cab82
YW
876
877 if (sd_device_get_sysattr_value(dev, "loop/backing_file", &s) >= 0) {
878 backing_file = strdup(s);
879 if (!backing_file)
880 return -ENOMEM;
881 }
4d2a9e3e
LP
882
883 backing_devno = info.lo_device;
884 backing_inode = info.lo_inode;
a8d8a619 885 }
b26c39ad 886
4f0ad43e 887 r = fd_get_diskseq(fd, &diskseq);
ffcb3324
YW
888 if (r < 0 && r != -EOPNOTSUPP)
889 return r;
890
22ee78a8
LP
891 uint32_t sector_size;
892 r = blockdev_get_sector_size(fd, &sector_size);
893 if (r < 0)
894 return r;
895
c961a8c6
LP
896 uint64_t device_size;
897 r = blockdev_get_device_size(fd, &device_size);
898 if (r < 0)
899 return r;
900
4f0ad43e
YW
901 r = sd_device_get_devnum(dev, &devnum);
902 if (r < 0)
903 return r;
7f52206a 904
4f0ad43e 905 r = sd_device_get_devname(dev, &s);
cc5bae6c
YW
906 if (r < 0)
907 return r;
908
4f0ad43e
YW
909 node = strdup(s);
910 if (!node)
cc5bae6c 911 return -ENOMEM;
9dabc4fd
LP
912
913 d = new(LoopDevice, 1);
914 if (!d)
915 return -ENOMEM;
916
917 *d = (LoopDevice) {
36d5eb0b 918 .n_ref = 1,
a8d8a619 919 .fd = TAKE_FD(fd),
7f52206a 920 .lock_fd = TAKE_FD(lock_fd),
b26c39ad 921 .nr = nr,
4f0ad43e
YW
922 .node = TAKE_PTR(node),
923 .dev = sd_device_ref(dev),
e77cab82 924 .backing_file = TAKE_PTR(backing_file),
4d2a9e3e
LP
925 .backing_inode = backing_inode,
926 .backing_devno = backing_devno,
9dabc4fd 927 .relinquished = true, /* It's not ours, don't try to destroy it when this object is freed */
4f0ad43e 928 .devno = devnum,
ffcb3324 929 .diskseq = diskseq,
22ee78a8 930 .sector_size = sector_size,
c961a8c6 931 .device_size = device_size,
6bc20134 932 .created = false,
9dabc4fd
LP
933 };
934
935 *ret = d;
4f0ad43e
YW
936 return 0;
937}
938
939int loop_device_open_from_fd(
940 int fd,
941 int open_flags,
942 int lock_op,
943 LoopDevice **ret) {
944
945 _cleanup_(sd_device_unrefp) sd_device *dev = NULL;
946 int r;
947
8e398254 948 r = block_device_new_from_fd(ASSERT_FD(fd), 0, &dev);
4f0ad43e
YW
949 if (r < 0)
950 return r;
951
952 return loop_device_open(dev, open_flags, lock_op, ret);
953}
954
955int loop_device_open_from_path(
956 const char *path,
957 int open_flags,
958 int lock_op,
959 LoopDevice **ret) {
960
961 _cleanup_(sd_device_unrefp) sd_device *dev = NULL;
962 int r;
963
964 assert(path);
965
966 r = block_device_new_from_path(path, 0, &dev);
967 if (r < 0)
968 return r;
969
970 return loop_device_open(dev, open_flags, lock_op, ret);
9dabc4fd
LP
971}
972
f1443709
LP
973static int resize_partition(int partition_fd, uint64_t offset, uint64_t size) {
974 char sysfs[STRLEN("/sys/dev/block/:/partition") + 2*DECIMAL_STR_MAX(dev_t) + 1];
ca822829 975 _cleanup_free_ char *buffer = NULL;
f1443709 976 uint64_t current_offset, current_size, partno;
254d1313 977 _cleanup_close_ int whole_fd = -EBADF;
f1443709
LP
978 struct stat st;
979 dev_t devno;
980 int r;
981
f1443709
LP
982 /* Resizes the partition the loopback device refer to (assuming it refers to one instead of an actual
983 * loopback device), and changes the offset, if needed. This is a fancy wrapper around
984 * BLKPG_RESIZE_PARTITION. */
985
8e398254 986 if (fstat(ASSERT_FD(partition_fd), &st) < 0)
f1443709
LP
987 return -errno;
988
989 assert(S_ISBLK(st.st_mode));
990
ed13feff 991 xsprintf(sysfs, "/sys/dev/block/" DEVNUM_FORMAT_STR "/partition", DEVNUM_FORMAT_VAL(st.st_rdev));
f1443709
LP
992 r = read_one_line_file(sysfs, &buffer);
993 if (r == -ENOENT) /* not a partition, cannot resize */
994 return -ENOTTY;
995 if (r < 0)
996 return r;
997 r = safe_atou64(buffer, &partno);
998 if (r < 0)
999 return r;
1000
ed13feff 1001 xsprintf(sysfs, "/sys/dev/block/" DEVNUM_FORMAT_STR "/start", DEVNUM_FORMAT_VAL(st.st_rdev));
f1443709
LP
1002
1003 buffer = mfree(buffer);
1004 r = read_one_line_file(sysfs, &buffer);
1005 if (r < 0)
1006 return r;
1007 r = safe_atou64(buffer, &current_offset);
1008 if (r < 0)
1009 return r;
1010 if (current_offset > UINT64_MAX/512U)
1011 return -EINVAL;
1012 current_offset *= 512U;
1013
c961a8c6
LP
1014 r = blockdev_get_device_size(partition_fd, &current_size);
1015 if (r < 0)
1016 return r;
f1443709
LP
1017
1018 if (size == UINT64_MAX && offset == UINT64_MAX)
1019 return 0;
1020 if (current_size == size && current_offset == offset)
1021 return 0;
1022
ed13feff 1023 xsprintf(sysfs, "/sys/dev/block/" DEVNUM_FORMAT_STR "/../dev", DEVNUM_FORMAT_VAL(st.st_rdev));
f1443709
LP
1024
1025 buffer = mfree(buffer);
1026 r = read_one_line_file(sysfs, &buffer);
1027 if (r < 0)
1028 return r;
7176f06c 1029 r = parse_devnum(buffer, &devno);
f1443709
LP
1030 if (r < 0)
1031 return r;
1032
ca822829 1033 whole_fd = r = device_open_from_devnum(S_IFBLK, devno, O_RDWR|O_CLOEXEC|O_NONBLOCK|O_NOCTTY, NULL);
f1443709
LP
1034 if (r < 0)
1035 return r;
1036
91e1ce1a
LP
1037 return block_device_resize_partition(
1038 whole_fd,
1039 partno,
1040 offset == UINT64_MAX ? current_offset : offset,
1041 size == UINT64_MAX ? current_size : size);
f1443709
LP
1042}
1043
c37878fc
LP
1044int loop_device_refresh_size(LoopDevice *d, uint64_t offset, uint64_t size) {
1045 struct loop_info64 info;
ff27ef4b 1046
9dabc4fd 1047 assert(d);
ff27ef4b 1048 assert(d->fd >= 0);
9dabc4fd 1049
f1443709
LP
1050 /* Changes the offset/start of the loop device relative to the beginning of the underlying file or
1051 * block device. If this loop device actually refers to a partition and not a loopback device, we'll
1052 * try to adjust the partition offsets instead.
1053 *
1054 * If either offset or size is UINT64_MAX we won't change that parameter. */
1055
f1443709
LP
1056 if (d->nr < 0) /* not a loopback device */
1057 return resize_partition(d->fd, offset, size);
1058
c37878fc
LP
1059 if (ioctl(d->fd, LOOP_GET_STATUS64, &info) < 0)
1060 return -errno;
1061
10c1b188
LP
1062#if HAVE_VALGRIND_MEMCHECK_H
1063 /* Valgrind currently doesn't know LOOP_GET_STATUS64. Remove this once it does */
1064 VALGRIND_MAKE_MEM_DEFINED(&info, sizeof(info));
1065#endif
1066
c37878fc
LP
1067 if (size == UINT64_MAX && offset == UINT64_MAX)
1068 return 0;
1069 if (info.lo_sizelimit == size && info.lo_offset == offset)
1070 return 0;
1071
1072 if (size != UINT64_MAX)
1073 info.lo_sizelimit = size;
1074 if (offset != UINT64_MAX)
1075 info.lo_offset = offset;
1076
7c248223 1077 return RET_NERRNO(ioctl(d->fd, LOOP_SET_STATUS64, &info));
9dabc4fd 1078}
441ec804
LP
1079
1080int loop_device_flock(LoopDevice *d, int operation) {
7f52206a 1081 assert(IN_SET(operation & ~LOCK_NB, LOCK_UN, LOCK_SH, LOCK_EX));
441ec804
LP
1082 assert(d);
1083
7f52206a
LP
1084 /* When unlocking just close the lock fd */
1085 if ((operation & ~LOCK_NB) == LOCK_UN) {
1086 d->lock_fd = safe_close(d->lock_fd);
1087 return 0;
1088 }
1089
1090 /* If we had no lock fd so far, create one and lock it right-away */
1091 if (d->lock_fd < 0) {
8e398254 1092 d->lock_fd = open_lock_fd(ASSERT_FD(d->fd), operation);
7f52206a
LP
1093 if (d->lock_fd < 0)
1094 return d->lock_fd;
1095
1096 return 0;
1097 }
441ec804 1098
7f52206a
LP
1099 /* Otherwise change the current lock mode on the existing fd */
1100 return RET_NERRNO(flock(d->lock_fd, operation));
441ec804 1101}
8dbc208c
LP
1102
1103int loop_device_sync(LoopDevice *d) {
1104 assert(d);
1105
1106 /* We also do this implicitly in loop_device_unref(). Doing this explicitly here has the benefit that
1107 * we can check the return value though. */
1108
8e398254 1109 return RET_NERRNO(fsync(ASSERT_FD(d->fd)));
8dbc208c 1110}
d2430d50
LP
1111
1112int loop_device_set_autoclear(LoopDevice *d, bool autoclear) {
1113 struct loop_info64 info;
1114
1115 assert(d);
1116
8e398254 1117 if (ioctl(ASSERT_FD(d->fd), LOOP_GET_STATUS64, &info) < 0)
d2430d50
LP
1118 return -errno;
1119
1120 if (autoclear == FLAGS_SET(info.lo_flags, LO_FLAGS_AUTOCLEAR))
1121 return 0;
1122
1123 SET_FLAG(info.lo_flags, LO_FLAGS_AUTOCLEAR, autoclear);
1124
1125 if (ioctl(d->fd, LOOP_SET_STATUS64, &info) < 0)
1126 return -errno;
1127
1128 return 1;
1129}
999ac3e2
LP
1130
1131int loop_device_set_filename(LoopDevice *d, const char *name) {
1132 struct loop_info64 info;
1133
1134 assert(d);
1135
1136 /* Sets the .lo_file_name of the loopback device. This is supposed to contain the path to the file
1137 * backing the block device, but is actually just a free-form string you can pass to the kernel. Most
1138 * tools that actually care for the backing file path use the sysfs attribute file loop/backing_file
7a05926f
YW
1139 * which is a kernel generated string, subject to file system namespaces and such.
1140 *
1141 * .lo_file_name is useful since userspace can select it freely when creating a loopback block
1142 * device, and we can use it for /dev/disk/by-loop-ref/ symlinks, and similar, so that apps can
1143 * recognize their own loopback files. */
999ac3e2
LP
1144
1145 if (name && strlen(name) >= sizeof(info.lo_file_name))
1146 return -ENOBUFS;
1147
8e398254 1148 if (ioctl(ASSERT_FD(d->fd), LOOP_GET_STATUS64, &info) < 0)
999ac3e2
LP
1149 return -errno;
1150
1151 if (strneq((char*) info.lo_file_name, strempty(name), sizeof(info.lo_file_name)))
1152 return 0;
1153
1154 if (name) {
1155 strncpy((char*) info.lo_file_name, name, sizeof(info.lo_file_name)-1);
1156 info.lo_file_name[sizeof(info.lo_file_name)-1] = 0;
1157 } else
1158 memzero(info.lo_file_name, sizeof(info.lo_file_name));
1159
1160 if (ioctl(d->fd, LOOP_SET_STATUS64, &info) < 0)
1161 return -errno;
1162
1163 return 1;
1164}