diff options
author | Andrew Cady <d@jerkface.net> | 2016-04-29 14:36:26 -0400 |
---|---|---|
committer | Andrew Cady <d@jerkface.net> | 2016-04-29 14:36:30 -0400 |
commit | 153d299a41b9be4e15dab1ca29bb93a74bd2445d (patch) | |
tree | 96fbfbe7c64f0b3f02f3d755e2b129917785bb98 /src/initrd/mdadm-dup.sh | |
parent | 5f41fb879ca830e5ad3345878e59072f3d6573bc (diff) |
fix paths (in progress)
Diffstat (limited to 'src/initrd/mdadm-dup.sh')
-rw-r--r-- | src/initrd/mdadm-dup.sh | 217 |
1 files changed, 217 insertions, 0 deletions
diff --git a/src/initrd/mdadm-dup.sh b/src/initrd/mdadm-dup.sh new file mode 100644 index 0000000..70163a5 --- /dev/null +++ b/src/initrd/mdadm-dup.sh | |||
@@ -0,0 +1,217 @@ | |||
1 | LoSetup() | ||
2 | { | ||
3 | local losetup_binary="$(which LoSetup)" | ||
4 | if [ "$losetup_binary" ]; then | ||
5 | "$losetup_binary" "$@" | ||
6 | else | ||
7 | losetup "$@" | ||
8 | fi | ||
9 | } | ||
10 | |||
11 | dm_snapshot() | ||
12 | { | ||
13 | # TODO: eliminate duplication; this function exists elsewhere in a less generalized form | ||
14 | local ro_file rw_file cutoff_size | ||
15 | ro_file=$1 | ||
16 | rw_file=$2 | ||
17 | cutoff_size=$3 | ||
18 | |||
19 | local ro_dev rw_dev size new_dev_name persist chunksize | ||
20 | |||
21 | if [ -b "$ro_file" ]; | ||
22 | then ro_dev=$ro_file | ||
23 | else ro_dev=$(LoSetup -r -f --show "$ro_file") || return | ||
24 | fi | ||
25 | |||
26 | if [ -b "$rw_file" ]; | ||
27 | then rw_dev=$rw_file | ||
28 | else rw_dev=$(LoSetup -f --show "$rw_file") || return | ||
29 | fi | ||
30 | |||
31 | if [ "$cutoff_size" -a "$cutoff_size" -gt 0 ]; then | ||
32 | size=$cutoff_size | ||
33 | else | ||
34 | size=$(blockdev --getsz "$ro_dev") || return | ||
35 | fi | ||
36 | |||
37 | new_dev_name=${ro_dev##*/} | ||
38 | persist=p | ||
39 | chunksize=16 | ||
40 | dmsetup create "$new_dev_name" --table "0 $size snapshot $ro_dev $rw_dev $persist $chunksize" || return | ||
41 | wait_for_dm_device /dev/mapper/"$new_dev_name" | ||
42 | echo /dev/mapper/"$new_dev_name" | ||
43 | } | ||
44 | |||
45 | dm_snapshot_teardown() | ||
46 | { | ||
47 | local dev="$1" | ||
48 | case "$dev" in | ||
49 | /dev/dm-*) | ||
50 | dmsetup table "$dev" | ( | ||
51 | read _ _ snapshot ro_dev rw_dev _ crypt_dev _ | ||
52 | case "$snapshot" in | ||
53 | snapshot) | ||
54 | dmsetup remove "$dev" || exit 1 | ||
55 | # errors ignored because the loop dev can be configured to be | ||
56 | # automatically removed upon disuse | ||
57 | losetup -d /dev/block/"$rw_dev" || true | ||
58 | eject /dev/block/"$ro_dev" || true | ||
59 | ;; | ||
60 | crypt) | ||
61 | cryptsetup remove "$dev" || exit 1 | ||
62 | losetup -d /dev/block/"$crypt_dev" || true | ||
63 | ;; | ||
64 | esac | ||
65 | ) || return | ||
66 | ;; | ||
67 | *) return 1 ;; | ||
68 | esac | ||
69 | } | ||
70 | |||
71 | wait_for_dm_device() | ||
72 | { | ||
73 | # TODO: improve | ||
74 | while ! [ -e "$1" ]; do | ||
75 | sleep 1 | ||
76 | done | ||
77 | } | ||
78 | |||
79 | dup_mount_cdrom() | ||
80 | { | ||
81 | local cdrom_dev="$1" mountpoint="$2" | ||
82 | |||
83 | local sectors md_dev=/dev/md55 cdrom_rw_file=/"${cdrom_dev##*/}".rw | ||
84 | |||
85 | sectors=$(get_cdrom_sizelimit "$cdrom_dev") || return | ||
86 | |||
87 | # TODO: do we even need this backing file? We do need to trick mdadm into | ||
88 | # thinking that this is a RW device, but previously we got away with just | ||
89 | # creating a loopback device. | ||
90 | dd if=/dev/zero of="$cdrom_rw_file" bs=1K count=32 || return | ||
91 | cdrom_rw_dev=$(dm_snapshot "$cdrom_dev" "$cdrom_rw_file" "$sectors") || return | ||
92 | mdadm_dup "$cdrom_rw_dev" "$md_dev" "$sectors" || return | ||
93 | mount -t iso9660 -r $md_dev "$mountpoint" | ||
94 | } | ||
95 | |||
96 | get_cdrom_sizelimit() | ||
97 | { | ||
98 | # returns 512-byte sectors | ||
99 | local dev="$1" sectors | ||
100 | sectors=$(blockdev --getsz "$dev") || return | ||
101 | |||
102 | # Check if we can read the last 8 sectors. With a TAO CDROM, we can't -- | ||
103 | # these sectors are faux, and not part of the ISO fs. If mdadm is allowed to | ||
104 | # read them, it will mark the device failed. | ||
105 | if dd count=2 if="$dev" bs=2048 skip=$((sectors/4 - 2)) of=/dev/null 2>/dev/null; then | ||
106 | echo $sectors | ||
107 | else | ||
108 | echo $((sectors - 8)) | ||
109 | fi | ||
110 | } | ||
111 | |||
112 | mdadm_dup() | ||
113 | { | ||
114 | local input_dev="$1" md_name="$2" sectors="$3" | ||
115 | |||
116 | mdadm --build "$md_name" "${sectors:+--size=$((sectors / 2))}" \ | ||
117 | --level=1 --raid-devices=1 --force --write-mostly "$input_dev" || return | ||
118 | } | ||
119 | |||
120 | mdadm_subdevices() | ||
121 | { | ||
122 | local md_dev="$1" | ||
123 | mdadm -D "$md_dev" -Y | sed -ne 's/^MD_DEVICE_.*_DEV=//p' | ||
124 | } | ||
125 | |||
126 | cryptsetup_temp() | ||
127 | { | ||
128 | local sectors="$1" cryptname="$2" temp_file="$3" parms=$- secret | ||
129 | set +x | ||
130 | # Add 4096 sectors for LUKS header | ||
131 | truncate -s $(((sectors + 4096) * 512)) "$temp_file" || return | ||
132 | cleartext_dev=$(LoSetup -f --show "$temp_file") || return | ||
133 | secret="$(head -c256 /dev/urandom)" || return | ||
134 | printf %s "$secret" | | ||
135 | cryptsetup luksFormat "$cleartext_dev" - || return | ||
136 | printf %s "$secret" | | ||
137 | cryptsetup --key-file - luksOpen "$cleartext_dev" "$cryptname" || return | ||
138 | unset secret | ||
139 | set "$parms" | ||
140 | |||
141 | wait_for_dm_device /dev/mapper/"$cryptname" | ||
142 | rm "$temp_file" | ||
143 | echo /dev/mapper/"$cryptname" | ||
144 | } | ||
145 | |||
146 | mdadm_copy_eject_crypt() | ||
147 | { | ||
148 | local md_dev="$1" temp_file="$2" | ||
149 | |||
150 | [ -b "$md_dev" ] || return | ||
151 | |||
152 | local output_dev sectors | ||
153 | |||
154 | old_subdev=$(mdadm_subdevices "$md_dev"|head -n1) || return | ||
155 | [ -b "$old_subdev" ] || return | ||
156 | # TODO: truncate to the ISO fs size if the device is larger | ||
157 | sectors=$(blockdev --getsz "$md_dev") || return | ||
158 | |||
159 | output_dev=$(cryptsetup_temp "$sectors" samizdatiso "$temp_file") || return | ||
160 | |||
161 | mdadm "$md_dev" --add "$output_dev" || return | ||
162 | mdadm "$md_dev" --grow -n2 || return | ||
163 | |||
164 | mdadm_wait_remove "$md_dev" "$old_subdev" || return | ||
165 | |||
166 | mdadm "$md_dev" --grow -n1 --force || return | ||
167 | dm_snapshot_teardown "$old_subdev" | ||
168 | } | ||
169 | |||
170 | mdadm_copy_eject() | ||
171 | { | ||
172 | local md_dev="$1" output_file="$2" | ||
173 | |||
174 | [ -b "$md_dev" ] || return | ||
175 | [ ! -e "$output_file" ] || return | ||
176 | |||
177 | local output_dev sectors | ||
178 | |||
179 | old_subdev=$(mdadm_subdevices "$md_dev"|head -n1) || return | ||
180 | [ -b "$old_subdev" ] || return | ||
181 | sectors=$(blockdev --getsz "$md_dev") || return | ||
182 | |||
183 | truncate -s $((sectors * 512)) "$output_file" || return | ||
184 | output_dev=$(LoSetup -f --show "$output_file") || return | ||
185 | |||
186 | mdadm "$md_dev" --add "$output_dev" || return | ||
187 | mdadm "$md_dev" --grow -n2 || return | ||
188 | |||
189 | mdadm_wait_remove "$md_dev" "$old_subdev" || return | ||
190 | |||
191 | mdadm "$md_dev" --grow -n1 --force || return | ||
192 | dm_snapshot_teardown "$old_subdev" | ||
193 | } | ||
194 | |||
195 | mdadm_wait_remove() | ||
196 | { | ||
197 | # We should perhaps use mdadm --monitor's RebuildFinished event. | ||
198 | |||
199 | local dev="$1" disk="$2" tries | ||
200 | if ! mdadm --wait "$dev"; then | ||
201 | tries=1000 | ||
202 | while ! mdadm --detail --test "$dev"; do | ||
203 | [ $tries -gt 0 ] || return 1 | ||
204 | sleep 1 | ||
205 | tries=$((tries-1)) | ||
206 | done | ||
207 | fi | ||
208 | |||
209 | mdadm "$dev" --fail "$disk" || return 1 | ||
210 | tries=100 | ||
211 | while ! mdadm "$dev" --remove "$disk"; do | ||
212 | [ $tries -gt 0 ] || return 1 | ||
213 | sleep 1 | ||
214 | tries=$((tries-1)) | ||
215 | done | ||
216 | return 0 | ||
217 | } | ||