1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
|
From 3947abeeb92069705229bff681b28d8ac78aedaa Mon Sep 17 00:00:00 2001
From: Colin Watson <cjwatson@ubuntu.com>
Date: Fri, 28 Mar 2014 17:10:16 +0000
Subject: udev handling
Run udevadm settle around partition table rereads, to avoid races.
This should be replaced by a proper completion-notification mechanism
between the kernel and udev.
Forwarded: no
Last-Update: 2019-10-11
Patch-Name: udevadm-settle.patch
---
libparted/arch/linux.c | 56 ++++++++++++++++++++++++++++++++++++++++--
1 file changed, 54 insertions(+), 2 deletions(-)
diff --git a/libparted/arch/linux.c b/libparted/arch/linux.c
index ccbba865..f6f72360 100644
--- a/libparted/arch/linux.c
+++ b/libparted/arch/linux.c
@@ -26,6 +26,7 @@
#include <parted/fdasd.h>
#endif
+#include <stdlib.h>
#include <ctype.h>
#include <errno.h>
#include <fcntl.h>
@@ -3250,10 +3251,52 @@ _have_blkpg ()
return have_blkpg = kver >= KERNEL_VERSION (2,4,0) ? 1 : 0;
}
+static int
+_chrooted ()
+{
+ static int cached = -1;
+ struct stat root, init_root;
+
+ if (cached != -1)
+ return cached;
+
+ if (stat ("/", &root) || stat ("/proc/1/root", &init_root))
+ /* We can't tell, but are unlikely to be able to tell in the
+ * future either.
+ */
+ cached = 0;
+ else if (root.st_dev == init_root.st_dev &&
+ root.st_ino == init_root.st_ino)
+ /* / has the same dev/ino as /sbin/init's root, so we're not
+ * in a chroot.
+ */
+ cached = 0;
+ else
+ /* We must be in a chroot. */
+ cached = 1;
+
+ return cached;
+}
+
/* Return nonzero upon success, 0 if something fails. */
static int
linux_disk_commit (PedDisk* disk)
{
+ int ret = 1;
+
+ /* Modern versions of udev may notice the write activity on
+ * partition devices caused by _flush_cache, and may decide to
+ * synthesise some change events as a result. These may in turn run
+ * programs that open partition devices, which will race with us
+ * trying to remove those devices. To avoid this, we need to wait
+ * until udevd has finished processing its event queue.
+ * TODO: for upstream submission, this should check whether udevadm
+ * exists on $PATH.
+ */
+ if (!_chrooted () && system ("udevadm settle") != 0) {
+ /* ignore failures */
+ }
+
if (disk->dev->type != PED_DEVICE_FILE) {
/* We now require BLKPG support. If this assertion fails,
@@ -3263,10 +3306,19 @@ linux_disk_commit (PedDisk* disk)
assert (_have_blkpg ());
if (!_disk_sync_part_table (disk))
- return 0;
+ ret = 0;
}
- return 1;
+ /* Now we wait for udevd to finish creating device nodes based on
+ * the above activity, so that callers can reliably use them.
+ * TODO: for upstream submission, this should check whether udevadm
+ * exists on $PATH.
+ */
+ if (!_chrooted () && system ("udevadm settle") != 0) {
+ /* ignore failures */
+ }
+
+ return ret;
}
#if USE_BLKID
|