From: Lennart Poettering Date: Tue, 21 May 2019 16:13:55 +0000 (+0200) Subject: loop-util: invoke LOOP_CTL_GET_FREE in a loop X-Git-Tag: v243-rc1~394 X-Git-Url: http://git.ipfire.org/cgi-bin/gitweb.cgi?a=commitdiff_plain;h=0f6519d43c37d5a855a6c9be52148908c15c3c9d;p=thirdparty%2Fsystemd.git loop-util: invoke LOOP_CTL_GET_FREE in a loop if we don't call it in a loop the device it tells us to open might already be gone, taken by somebody else racing against us. Hence try a few times. --- diff --git a/src/shared/loop-util.c b/src/shared/loop-util.c index bf426eb8bca..22525d5c633 100644 --- a/src/shared/loop-util.c +++ b/src/shared/loop-util.c @@ -18,6 +18,7 @@ int loop_device_make(int fd, int open_flags, LoopDevice **ret) { _cleanup_close_ int control = -1, loop = -1; _cleanup_free_ char *loopdev = NULL; + unsigned n_attempts = 0; struct stat st; LoopDevice *d; int nr, r; @@ -60,19 +61,31 @@ int loop_device_make(int fd, int open_flags, LoopDevice **ret) { if (control < 0) return -errno; - nr = ioctl(control, LOOP_CTL_GET_FREE); - if (nr < 0) - return -errno; + /* Loop around LOOP_CTL_GET_FREE, since at the moment we attempt to open the returned device it might + * be gone already, taken by somebody else racing against us. */ + for (;;) { + nr = ioctl(control, LOOP_CTL_GET_FREE); + if (nr < 0) + return -errno; - if (asprintf(&loopdev, "/dev/loop%i", nr) < 0) - return -ENOMEM; + if (asprintf(&loopdev, "/dev/loop%i", nr) < 0) + return -ENOMEM; - loop = open(loopdev, O_CLOEXEC|O_NONBLOCK|O_NOCTTY|open_flags); - if (loop < 0) - return -errno; + loop = open(loopdev, O_CLOEXEC|O_NONBLOCK|O_NOCTTY|open_flags); + if (loop < 0) + return -errno; + if (ioctl(loop, LOOP_SET_FD, fd) < 0) { + if (errno != EBUSY) + return -errno; - if (ioctl(loop, LOOP_SET_FD, fd) < 0) - return -errno; + if (++n_attempts >= 64) /* Give up eventually */ + return -EBUSY; + } else + break; + + loopdev = mfree(loopdev); + loop = safe_close(loop); + } if (ioctl(loop, LOOP_SET_STATUS64, &info) < 0) return -errno;