У меня есть несколько машин с Ubuntu как с SSD, так и с традиционным жестким диском, причем жесткий диск используется лишь изредка.
В целях снижения шума, нагрева и энергопотребления, а также для продления срока службы жесткого диска я бы хотел отключать его при каждой загрузке и выводить из спящего режима только при необходимости.
hdparm -y
(или -Y
) отлично работает в командной строке и делает именно то, что мне нужно.
Но написание службы systemd для выполнения hdparm
не работает. Точнее: это работает, и диск действительно засыпает (как показывает отладка), но он тут же снова просыпается (и остается бодрствующим), потому что в системе есть что-то, что обращается к жестким дискам (и, таким образом, пробуждает их) в самом конце процесса загрузки systemd.
Так как же мне включить hdparm -y
процесс загрузки достаточно поздно, чтобы за ним больше не последовал никакой другой процесс?
Моя последняя догадка — изменить цель systemd по умолчанию с графической на новую (sleepydisks), которая затем будет зависеть от предыдущей графической цели.
Но разве нет более простого и понятного способа отключения дисков?
С уважением
решение1
Откройте утилиту Disks и настройте диск на автоматическое остановку вращения после 10 минут бездействия.
решение2
Используйте опцию -S команды hdparm. Это устанавливает тайм-аут замедления вращения, который должен поддерживаться приводом.
$ hdparm --help
-S Set standby (spindown) timeout
$ hdparm -S 60 /dev/sdd
/dev/sdd:
setting standby to 60 (5 minutes)
Обратите внимание, что это заглавная буква «S».
решение3
Я использую следующий скрипт, чтобы «заставить» простаивающие диски начать вращение, где под «заставить» я подразумеваю прямую команду диску, поскольку многие диски WD имеют проблемы с самостоятельным вращением. Возможно, вы сможете расширить это для своих нужд. Скрипт хорошо прокомментирован, так что вы сможете довольно легко его изменить. Я вызываю его каждые cron
20 минут, корректирую по мере необходимости:
spindown.sh
#!/usr/bin/env bash
logger "[SPINDOWN] Checking disk activity and spinning down idles..."
# Exit during maintenance
if pgrep snapraid > /dev/null; then
logger "[SPINDOWN] Detected snapRAID maintenance. Exiting.";
exit 0
fi
# Specify any drives you want to ignore; separate multiple drives by spaces; e.g. "sda sdb"
IGNORE_DRIVES=""
PATH='/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin'
# Check for idle disks and spin them down unless smartd is running tests
# Create a file on the ramdisk and cycle it to test for disk activity
( if [ ! -f /dev/shm/diskstats_1 ]; then
logger "[SPINDOWN] Creating initial state file on ramdisk";
touch /dev/shm/diskstats_1;
fi ;
logger "[SPINDOWN] Updating state files on ramdisk";
mv /dev/shm/diskstats_1 /dev/shm/diskstats_2;
cat /proc/diskstats > /dev/shm/diskstats_1 ) > /dev/null 2>&1
# Find all removable USB drives, so we can ignore them later,
# see http://superuser.com/a/465953
REMOVABLE_DRIVES=""
for _device in /sys/block/*/device; do
if echo $(readlink -f "$_device")|egrep -q "usb"; then
_disk=$(echo "$_device" | cut -f4 -d/)
REMOVABLE_DRIVES="$REMOVABLE_DRIVES $_disk"
fi
done
# Append detected removable drives to manually ignored drives
IGNORE_DRIVES="$IGNORE_DRIVES $REMOVABLE_DRIVES"
# Loop through all the array disks and spin down the idle disks. Will find all drives sda > sdz AND sdaa > sdaz...
logger "[SPINDOWN] Looping through drives to detect idle state"
for disk in `find /dev/ -regex '/dev/sd[a-z]+' | cut -d/ -f3`
do
# Skip removable USB drives and those the user wants to ignore
if [[ $IGNORE_DRIVES =~ $disk ]]; then
continue
fi
# Skip SSDs
if [[ $(cat /sys/block/$disk/queue/rotational) -eq 0 ]]; then
continue
fi
# Check if drive exists
if [ -e /dev/$disk ]; then
logger "[SPINDOWN] Checking drive $disk"
# Check if drive is currently spinning
if [ "$(smartctl -i -n standby /dev/$disk | grep "ACTIVE or IDLE")" ]; then
logger "[SPINDOWN] Disk $disk is ACTIVE or IDLE. Checking for selftest"
# Check if smartctl is currently not running a self test
if [ $(smartctl -a /dev/$disk | grep -c "Self-test routine in progress") = 0 ]; then
logger "[SPINDOWN] Disk $disk is not running a self test. Checking activity."
# Check if drive has been non idle since last run
if [ "$(diff /dev/shm/diskstats_1 /dev/shm/diskstats_2 | grep $disk )" = "" ]; then
logger "[SPINDOWN] Spinning down /dev/$disk `df -h | grep /dev/$disk | rev | cut -d ' ' -f 1 | rev`"
hdparm -y /dev/$disk
fi
else
logger "[SPINDOWN] /dev/$disk is running Self-test routine. Skipping."
fi
fi
fi
done
Что касается конкретного вопроса, как это сделать с помощью systemd? ... создайте файл таймера, который вызывает этот скрипт (Ссылка:https://wiki.archlinux.org/index.php/Systemd/Таймеры)