2013-03-20 05:27:32 +00:00
|
|
|
#!/bin/bash
|
|
|
|
|
2016-05-16 02:36:17 +00:00
|
|
|
shopt -s nullglob
|
|
|
|
|
2016-05-17 20:41:51 +00:00
|
|
|
|
2014-07-05 14:13:08 +00:00
|
|
|
export LC_CTYPE=en_US.UTF-8
|
2013-03-20 05:27:32 +00:00
|
|
|
NAME=${DEVNAME#/dev/}
|
2014-07-05 14:13:08 +00:00
|
|
|
DESC="`echo "${ID_MODEL} (${ID_FS_LABEL})" | iconv -f utf8 -t ascii//TRANSLIT`"
|
2013-03-20 05:27:32 +00:00
|
|
|
SIZE=$[ $(cat /sys/$DEVPATH/size) * 512 ]
|
|
|
|
MODE=w
|
2013-06-07 03:21:41 +00:00
|
|
|
QDB_KEY="/qubes-block-devices/$NAME"
|
2013-03-20 05:27:32 +00:00
|
|
|
|
|
|
|
xs_remove() {
|
|
|
|
if [ "$QUBES_EXPOSED" == "1" ]; then
|
2016-01-14 04:01:27 +00:00
|
|
|
qubesdb-rm "$QDB_KEY/"
|
2013-06-07 03:21:41 +00:00
|
|
|
qubesdb-write /qubes-block-devices ''
|
2013-03-20 05:27:32 +00:00
|
|
|
fi
|
|
|
|
echo QUBES_EXPOSED=0
|
|
|
|
}
|
|
|
|
|
2016-05-16 02:36:17 +00:00
|
|
|
is_used() {
|
|
|
|
local sys_devpath=$1
|
2016-07-15 16:15:41 +00:00
|
|
|
# mounted; or enabled swap
|
|
|
|
if lsblk -dnr -o MOUNTPOINT "$DEVNAME" | grep -q .; then
|
2016-05-16 02:36:17 +00:00
|
|
|
return 0
|
|
|
|
fi
|
|
|
|
# part of other device-mapper
|
|
|
|
if [ -n "`ls -A $sys_devpath/holders 2> /dev/null`" ]; then
|
|
|
|
return 0
|
|
|
|
fi
|
|
|
|
# open device-mapper device
|
|
|
|
if [ -f "$sys_devpath/dm/name" ] && \
|
|
|
|
/sbin/dmsetup info "$(cat $sys_devpath/dm/name)" |\
|
|
|
|
grep -q "^Open count:.*[1-9]"; then
|
|
|
|
return 0
|
|
|
|
fi
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2016-05-17 20:41:51 +00:00
|
|
|
# communicate with xenstored through socket in dom0
|
|
|
|
# trying to access xenstore before xenstored is started, hang forever (in
|
|
|
|
# non-killable state), so better fail ('-s' in VM if /proc/xen isn't mounted
|
|
|
|
# yet) than hang dom0 boot
|
|
|
|
if [ ! -r /proc/xen/capabilities ] || grep -q control_d /proc/xen/capabilities; then
|
|
|
|
XENSTORE_LS="xenstore-ls -s"
|
|
|
|
else
|
|
|
|
XENSTORE_LS="xenstore-ls"
|
|
|
|
fi
|
2016-05-16 02:39:19 +00:00
|
|
|
is_attached() {
|
|
|
|
dev_hex=$(stat -c %t:%T /dev/$(basename $1))
|
2016-05-17 20:41:51 +00:00
|
|
|
$XENSTORE_LS backend/vbd | grep -q "physical-device = \"$dev_hex\""
|
2016-05-16 02:39:19 +00:00
|
|
|
}
|
|
|
|
|
2016-05-16 02:36:17 +00:00
|
|
|
# update info about parent devices, if any:
|
|
|
|
if [ -f /sys$DEVPATH/partition ]; then
|
|
|
|
parent=$(dirname $(readlink -f /sys$DEVPATH))
|
|
|
|
udevadm trigger \
|
|
|
|
--property-match=DEVPATH=/$(realpath --relative-to=/sys $parent)
|
2016-05-16 02:39:19 +00:00
|
|
|
# if parent device is already attached, skip its partitions
|
|
|
|
if is_attached $parent; then
|
|
|
|
xs_remove
|
|
|
|
exit 0
|
|
|
|
fi
|
2013-03-20 05:27:32 +00:00
|
|
|
fi
|
2016-05-16 02:36:17 +00:00
|
|
|
|
|
|
|
# and underlying devices of device-mapper (if any)
|
|
|
|
for dev in /sys$DEVPATH/slaves/*; do
|
|
|
|
udevadm trigger \
|
|
|
|
--property-match=DEVPATH=/$(realpath --relative-to=/sys $dev)
|
|
|
|
done
|
|
|
|
|
|
|
|
# then take care of this device:
|
|
|
|
|
|
|
|
# device itself is already used
|
|
|
|
if is_used /sys$DEVPATH; then
|
2014-06-07 02:53:32 +00:00
|
|
|
xs_remove
|
|
|
|
exit 0
|
|
|
|
fi
|
2016-05-16 02:36:17 +00:00
|
|
|
|
|
|
|
# or one of its partitions is used
|
2016-05-16 02:39:19 +00:00
|
|
|
# or already attached (prevent attaching both device and its partition(s) at
|
|
|
|
# the same time)
|
2016-05-16 02:36:17 +00:00
|
|
|
for part in /sys$DEVPATH/$NAME*; do
|
|
|
|
if [ -d $part ]; then
|
|
|
|
if is_used $part; then
|
|
|
|
xs_remove
|
|
|
|
exit 0
|
|
|
|
fi
|
2016-05-16 02:39:19 +00:00
|
|
|
if is_attached $part; then
|
|
|
|
xs_remove
|
|
|
|
exit 0
|
|
|
|
fi
|
2016-05-16 02:36:17 +00:00
|
|
|
fi
|
|
|
|
done
|
|
|
|
|
|
|
|
# or "empty" loop device
|
2013-03-20 05:27:32 +00:00
|
|
|
if [ "$MAJOR" -eq 7 -a ! -d /sys/$DEVPATH/loop ]; then
|
|
|
|
xs_remove
|
|
|
|
exit 0
|
|
|
|
fi
|
2016-05-16 02:36:17 +00:00
|
|
|
|
2015-04-03 09:14:15 +00:00
|
|
|
# ... and temporary devices used during VM startup
|
|
|
|
if [[ "$NAME" = 'loop'* ]] && \
|
|
|
|
[[ "`cat /sys/block/${NAME%p*}/loop/backing_file`" = \
|
2015-03-30 02:54:37 +00:00
|
|
|
'/var/lib/qubes/'*'/volatile.img' ]]; then
|
|
|
|
xs_remove
|
|
|
|
exit 0
|
|
|
|
fi
|
2013-03-20 05:27:32 +00:00
|
|
|
|
2013-10-01 02:19:14 +00:00
|
|
|
# Check if device is read-only
|
|
|
|
if [ "`cat /sys/$DEVPATH/ro`" -eq 1 ]; then
|
|
|
|
MODE=r
|
|
|
|
fi
|
|
|
|
|
2013-03-20 05:27:32 +00:00
|
|
|
# Special case for CD
|
|
|
|
if [ "$ID_TYPE" = "cd" ]; then
|
|
|
|
if [ "$ID_CDROM_MEDIA" != "1" ]; then
|
|
|
|
# Hide empty cdrom drive
|
|
|
|
xs_remove
|
|
|
|
exit 0
|
|
|
|
fi
|
|
|
|
MODE=r
|
|
|
|
fi
|
|
|
|
|
|
|
|
# Special description for loop devices
|
|
|
|
if [ -d /sys/$DEVPATH/loop ]; then
|
|
|
|
DESC=$(cat /sys/$DEVPATH/loop/backing_file)
|
|
|
|
fi
|
2014-06-07 02:53:32 +00:00
|
|
|
# and for device-mapper
|
|
|
|
if [ -n "$DM_NAME" ]; then
|
|
|
|
DESC="$DM_NAME"
|
|
|
|
fi
|
2013-03-20 05:27:32 +00:00
|
|
|
|
2013-06-07 03:21:41 +00:00
|
|
|
# The last one is meant to trigger watches
|
|
|
|
qubesdb-write \
|
|
|
|
"$QDB_KEY/desc" "$DESC" \
|
2014-12-11 05:06:57 +00:00
|
|
|
"$QDB_KEY/size" "$SIZE" \
|
|
|
|
"$QDB_KEY/mode" "$MODE" \
|
2013-06-07 03:21:41 +00:00
|
|
|
/qubes-block-devices ''
|
2013-03-20 05:27:32 +00:00
|
|
|
echo QUBES_EXPOSED=1
|
|
|
|
|
|
|
|
# Make sure that block backend is loaded
|
|
|
|
/sbin/modprobe xen-blkback 2> /dev/null || /sbin/modprobe blkbk
|