81509a780d02b9c61d2fa4e2881532237f44ebb4
[live-boot-grml.git] / scripts / boot / 9990-overlay.sh
1 #!/bin/sh
2
3 #set -e
4
5 setup_unionfs ()
6 {
7         image_directory="${1}"
8         rootmnt="${2}"
9         addimage_directory="${3}"
10
11         case ${UNIONTYPE} in
12                 aufs|unionfs|overlayfs)
13                         modprobe -q -b ${UNIONTYPE}
14
15                         if ! cut -f2 /proc/filesystems | grep -q "^${UNIONTYPE}\$" && [ -x /bin/unionfs-fuse ]
16                         then
17                                 echo "${UNIONTYPE} not available, falling back to unionfs-fuse."
18                                 echo "This might be really slow."
19
20                                 UNIONTYPE="unionfs-fuse"
21                         fi
22                         ;;
23         esac
24
25         case "${UNIONTYPE}" in
26                 unionfs-fuse)
27                         modprobe fuse
28                         ;;
29         esac
30
31         # run-init can't deal with images in a subdir, but we're going to
32         # move all of these away before it runs anyway.  No, we're not,
33         # put them in / since move-mounting them into / breaks mono and
34         # some other apps.
35
36         croot="/"
37
38         # Let's just mount the read-only file systems first
39         rofslist=""
40
41         if [ -z "${PLAIN_ROOT}" ]
42         then
43                 # Read image names from ${MODULE}.module if it exists
44                 if [ -e "${image_directory}/filesystem.${MODULE}.module" ]
45                 then
46                         for IMAGE in $(cat ${image_directory}/filesystem.${MODULE}.module)
47                         do
48                                 image_string="${image_string} ${image_directory}/${IMAGE}"
49                         done
50                 elif [ -e "${image_directory}/${MODULE}.module" ]
51                 then
52                         for IMAGE in $(cat ${image_directory}/${MODULE}.module)
53                         do
54                                 image_string="${image_string} ${image_directory}/${IMAGE}"
55                         done
56                 else
57                         # ${MODULE}.module does not exist, create a list of images
58                         for FILESYSTEM in squashfs ext2 ext3 ext4 xfs jffs2 dir
59                         do
60                                 for IMAGE in "${image_directory}"/*."${FILESYSTEM}"
61                                 do
62                                         if [ -e "${IMAGE}" ]
63                                         then
64                                                 image_string="${image_string} ${IMAGE}"
65                                         fi
66                                 done
67                         done
68
69                         if [ -n "${addimage_directory}" ] && [ -d "${addimage_directory}" ]
70                         then
71                                 for FILESYSTEM in squashfs ext2 ext3 ext4 xfs jffs2 dir
72                                 do
73                                         for IMAGE in "${addimage_directory}"/*."${FILESYSTEM}"
74                                         do
75                                                 if [ -e "${IMAGE}" ]
76                                                 then
77                                                         image_string="${image_string} ${IMAGE}"
78                                                 fi
79                                         done
80                                 done
81                         fi
82
83                         # Now sort the list
84                         image_string="$(echo ${image_string} | sed -e 's/ /\n/g' | sort )"
85                 fi
86
87                 [ -n "${MODULETORAMFILE}" ] && image_string="${image_directory}/$(basename ${MODULETORAMFILE})"
88
89                 mkdir -p "${croot}"
90
91                 for image in ${image_string}
92                 do
93                         imagename=$(basename "${image}")
94
95                         export image devname
96                         maybe_break live-realpremount
97                         log_begin_msg "Running /scripts/live-realpremount"
98                         run_scripts /scripts/live-realpremount
99                         log_end_msg
100
101                         if [ -d "${image}" ]
102                         then
103                                 # it is a plain directory: do nothing
104                                 rofslist="${image} ${rofslist}"
105                         elif [ -f "${image}" ]
106                         then
107                                 if losetup --help 2>&1 | grep -q -- "-r\b"
108                                 then
109                                         backdev=$(get_backing_device "${image}" "-r")
110                                 else
111                                         backdev=$(get_backing_device "${image}")
112                                 fi
113                                 fstype=$(get_fstype "${backdev}")
114
115                                 case "${fstype}" in
116                                         unknown)
117                                                 panic "Unknown file system type on ${backdev} (${image})"
118                                                 ;;
119
120                                         "")
121                                                 fstype="${imagename##*.}"
122                                                 log_warning_msg "Unknown file system type on ${backdev} (${image}), assuming ${fstype}."
123                                                 ;;
124                                 esac
125
126                                 case "${UNIONTYPE}" in
127                                         unionmount)
128                                                 mpoint="${rootmnt}"
129                                                 rofslist="${rootmnt} ${rofslist}"
130                                                 ;;
131
132                                         *)
133                                                 mpoint="${croot}/${imagename}"
134                                                 rofslist="${mpoint} ${rofslist}"
135                                                 ;;
136                                 esac
137
138                                 mkdir -p "${mpoint}"
139                                 log_begin_msg "Mounting \"${image}\" on \"${mpoint}\" via \"${backdev}\""
140                                 mount -t "${fstype}" -o ro,noatime "${backdev}" "${mpoint}" || panic "Can not mount ${backdev} (${image}) on ${mpoint}"
141                                 log_end_msg
142                         fi
143                 done
144         else
145                 # we have a plain root system
146                 mkdir -p "${croot}/filesystem"
147                 log_begin_msg "Mounting \"${image_directory}\" on \"${croot}/filesystem\""
148                 mount -t $(get_fstype "${image_directory}") -o ro,noatime "${image_directory}" "${croot}/filesystem" || \
149                         panic "Can not mount ${image_directory} on ${croot}/filesystem" && \
150                         rofslist="${croot}/filesystem ${rofslist}"
151                 # probably broken:
152                 mount -o bind ${croot}/filesystem $mountpoint
153                 log_end_msg
154         fi
155
156         # tmpfs file systems
157         touch /etc/fstab
158         mkdir -p /live/overlay
159         mount -t tmpfs tmpfs /live/overlay
160
161         # Looking for persistence devices or files
162         if [ -n "${PERSISTENCE}" ] && [ -z "${NOPERSISTENCE}" ]
163         then
164
165                 if [ -z "${QUICKUSBMODULES}" ]
166                 then
167                         # Load USB modules
168                         num_block=$(ls -l /sys/block | wc -l)
169                         for module in sd_mod uhci-hcd ehci-hcd ohci-hcd usb-storage
170                         do
171                                 modprobe -q -b ${module}
172                         done
173
174                         udevadm trigger
175                         udevadm settle
176
177                         # For some reason, udevsettle does not block in this scenario,
178                         # so we sleep for a little while.
179                         #
180                         # See https://bugs.launchpad.net/ubuntu/+source/casper/+bug/84591
181                         for timeout in 5 4 3 2 1
182                         do
183                                 sleep 1
184
185                                 if [ $(ls -l /sys/block | wc -l) -gt ${num_block} ]
186                                 then
187                                         break
188                                 fi
189                         done
190                 fi
191
192                 case "${PERSISTENCE_MEDIA}" in
193                         removable)
194                                 whitelistdev="$(removable_dev)"
195                                 ;;
196
197                         removable-usb)
198                                 whitelistdev="$(removable_usb_dev)"
199                                 ;;
200
201                         *)
202                                 whitelistdev=""
203                                 ;;
204                 esac
205
206                 if is_in_comma_sep_list overlay ${PERSISTENCE_METHOD}
207                 then
208                         overlays="${old_root_overlay_label} ${old_home_overlay_label} ${custom_overlay_label}"
209                 fi
210
211                 local overlay_devices=""
212                 for media in $(find_persistence_media "${overlays}" "${whitelistdev}")
213                 do
214                         media="$(echo ${media} | tr ":" " ")"
215
216                         case ${media} in
217                                 ${old_root_overlay_label}=*)
218                                         device="${media#*=}"
219                                         fix_backwards_compatibility ${device} / union
220                                         overlay_devices="${overlay_devices} ${device}"
221                                         ;;
222
223                                 ${old_home_overlay_label}=*)
224                                         device="${media#*=}"
225                                         fix_backwards_compatibility ${device} /home bind
226                                         overlay_devices="${overlay_devices} ${device}"
227                                         ;;
228
229                                 ${custom_overlay_label}=*)
230                                         device="${media#*=}"
231                                         overlay_devices="${overlay_devices} ${device}"
232                                         ;;
233                          esac
234                 done
235         elif [ -n "${NFS_COW}" ] && [ -z "${NOPERSISTENCE}" ]
236         then
237                 # check if there are any nfs options
238                 if echo ${NFS_COW} | grep -q ','
239                 then
240                         nfs_cow_opts="-o nolock,$(echo ${NFS_COW}|cut -d, -f2-)"
241                         nfs_cow=$(echo ${NFS_COW}|cut -d, -f1)
242                 else
243                         nfs_cow_opts="-o nolock"
244                         nfs_cow=${NFS_COW}
245                 fi
246
247                 if [ -n "${PERSISTENCE_READONLY}" ]
248                 then
249                         nfs_cow_opts="${nfs_cow_opts},nocto,ro"
250                 fi
251
252                 mac="$(get_mac)"
253                 if [ -n "${mac}" ]
254                 then
255                         cowdevice=$(echo ${nfs_cow} | sed "s/client_mac_address/${mac}/")
256                         cow_fstype="nfs"
257                 else
258                         panic "unable to determine mac address"
259                 fi
260         fi
261
262         if [ -z "${cowdevice}" ]
263         then
264                 cowdevice="tmpfs"
265                 cow_fstype="tmpfs"
266                 cow_mountopt="rw,noatime,mode=755"
267         fi
268
269         if [ "${UNIONTYPE}" != "unionmount" ]
270         then
271                 if [ -n "${PERSISTENCE_READONLY}" ] && [ "${cowdevice}" != "tmpfs" ]
272                 then
273                         mount -t tmpfs -o rw,noatime,mode=755 tmpfs "/live/overlay"
274                         root_backing="/live/persistence/$(basename ${cowdevice})-root"
275                         mkdir -p ${root_backing}
276                 else
277                         root_backing="/live/overlay"
278                 fi
279
280                 if [ "${cow_fstype}" = "nfs" ]
281                 then
282                         log_begin_msg \
283                                 "Trying nfsmount ${nfs_cow_opts} ${cowdevice} ${root_backing}"
284                         nfsmount ${nfs_cow_opts} ${cowdevice} ${root_backing} || \
285                                 panic "Can not mount ${cowdevice} (n: ${cow_fstype}) on ${root_backing}"
286                 else
287                         mount -t ${cow_fstype} -o ${cow_mountopt} ${cowdevice} ${root_backing} || \
288                                 panic "Can not mount ${cowdevice} (o: ${cow_fstype}) on ${root_backing}"
289                 fi
290         fi
291
292         rofscount=$(echo ${rofslist} |wc -w)
293
294         rofs=${rofslist%% }
295
296         if [ -n "${EXPOSED_ROOT}" ]
297         then
298                 if [ ${rofscount} -ne 1 ]
299                 then
300                         panic "only one RO file system supported with exposedroot: ${rofslist}"
301                 fi
302
303                 mount --bind ${rofs} ${rootmnt} || \
304                         panic "bind mount of ${rofs} failed"
305
306                 if [ -z "${SKIP_UNION_MOUNTS}" ]
307                 then
308                         cow_dirs='/var/tmp /var/lock /var/run /var/log /var/spool /home /var/lib/live'
309                 else
310                         cow_dirs=''
311                 fi
312         else
313                 cow_dirs="/"
314         fi
315
316         if [ "${cow_fstype}" != "tmpfs" ] && [ "${cow_dirs}" != "/" ] && [ "${UNIONTYPE}" = "unionmount" ]
317         then
318                 true # FIXME: Maybe it does, I don't really know.
319                 #panic "unionmount does not support subunions (${cow_dirs})."
320         fi
321
322         for dir in ${cow_dirs}; do
323                 unionmountpoint="${rootmnt}${dir}"
324                 mkdir -p ${unionmountpoint}
325                 if [ "${UNIONTYPE}" = "unionmount" ]
326                 then
327                         # FIXME: handle PERSISTENCE_READONLY
328                         unionmountopts="-t ${cow_fstype} -o noatime,union,${cow_mountopt} ${cowdevice}"
329                         mount_full $unionmountopts "${unionmountpoint}"
330                 else
331                         cow_dir="/live/overlay${dir}"
332                         rofs_dir="${rofs}${dir}"
333                         mkdir -p ${cow_dir}
334                         if [ -n "${PERSISTENCE_READONLY}" ] && [ "${cowdevice}" != "tmpfs" ]
335                         then
336                                 do_union ${unionmountpoint} ${cow_dir} ${root_backing} ${rofs_dir}
337                         else
338                                 do_union ${unionmountpoint} ${cow_dir} ${rofs_dir}
339                         fi
340                 fi || panic "mount ${UNIONTYPE} on ${unionmountpoint} failed with option ${unionmountopts}"
341         done
342
343         # Correct the permissions of /:
344         chmod 0755 "${rootmnt}"
345
346         # Correct the permission of /tmp:
347         if [ -d "${rootmnt}/tmp" ]
348         then
349                 chmod 1777 "${rootmnt}"/tmp
350         fi
351
352         live_rofs_list=""
353         for d in ${rofslist}
354         do
355                 live_rofs="/live/rofs/${d##*/}"
356                 live_rofs_list="${live_rofs_list} ${live_rofs}"
357                 mkdir -p "${live_rofs}"
358                 case d in
359                         *.dir)
360                                 # do nothing # mount -o bind "${d}" "${live_rofs}"
361                                 ;;
362                         *)
363                                 case "${UNIONTYPE}" in
364                                         unionfs-fuse)
365                                                 mount -o bind "${d}" "${live_rofs}"
366                                                 ;;
367
368                                         *)
369                                                 mount -o move "${d}" "${live_rofs}"
370                                                 ;;
371                                 esac
372                                 ;;
373                 esac
374         done
375
376         # Adding custom persistence
377         if [ -n "${PERSISTENCE}" ] && [ -z "${NOPERSISTENCE}" ]
378         then
379                 local custom_mounts="/tmp/custom_mounts.list"
380                 rm -rf ${custom_mounts} 2> /dev/null
381
382                 # Gather information about custom mounts from devies detected as overlays
383                 get_custom_mounts ${custom_mounts} ${overlay_devices}
384
385                 [ -n "${DEBUG}" ] && cp ${custom_mounts} "/live/persistence"
386
387                 # Now we do the actual mounting (and symlinking)
388                 local used_overlays=""
389                 used_overlays=$(activate_custom_mounts ${custom_mounts})
390                 rm ${custom_mounts}
391
392                 # Close unused overlays (e.g. due to missing $persistence_list)
393                 for overlay in ${overlay_devices}
394                 do
395                         if echo ${used_overlays} | grep -qve "^\(.* \)\?${device}\( .*\)\?$"
396                         then
397                                 close_persistence_media ${overlay}
398                         fi
399                 done
400         fi
401
402         # move all mountpoints to root filesystem
403         for _DIRECTORY in rofs persistence
404         do
405                 if [ -d "/live/${_DIRECTORY}" ]
406                 then
407                         mkdir -p "${rootmnt}/lib/live/${_DIRECTORY}"
408
409                         for _MOUNT in $(ls /live/${_DIRECTORY})
410                         do
411                                 mkdir -p "${rootmnt}/lib/live/${_DIRECTORY}/${_MOUNT}"
412                                 mount -o move "/live/${_DIRECTORY}/${_MOUNT}" "${rootmnt}/lib/live/${_DIRECTORY}/${_MOUNT}" > /dev/null 2>&1 || \
413                                         mount -o bind "/live/${_DIRECTORY}/${_MOUNT}" "${rootmnt}/lib/live/${_DIRECTORY}/${_MOUNT}" || \
414                                         log_warning_msg "W: failed to mount /live/${_DIRECTORY}/${_MOUNT} to ${rootmnt}/lib/live/${_DIRECTORY}/${_MOUNT}"
415                         done
416                 fi
417         done
418
419         mkdir -p "${rootmnt}/lib/live/overlay"
420         mount -o move /live/overlay "${rootmnt}/lib/live/overlay" > /dev/null 2>&1 || \
421                 mount -o bind /live/overlay "${rootmnt}/lib/live/overlay" || \
422                 log_warning_msg "W: failed to mount /live/overlay to ${rootmnt}/lib/live/overlay"
423
424         # ensure that a potentially stray tmpfs gets removed
425         # otherways, initramfs-tools is unable to remove /live
426         # and fails to boot
427         umount /live/overlay || true
428 }