mirror of
https://github.com/Spearfoot/disk-burnin-and-testing.git
synced 2025-12-05 23:53:31 +00:00
Added -x option to control the badblocks -e option, allowing extended testing. Added smartctl to the list of dependencies. Changed disk type detection so that we assume all drives are mechanical drives unless they explicitly return 'Solid State Drive' for Rotational Rate. Removed datestamp from every line of log output, only emitting it in log headers. Minor reformatting.
609 lines
19 KiB
Bash
Executable File
609 lines
19 KiB
Bash
Executable File
#!/bin/sh
|
|
################################################################################
|
|
#
|
|
# disk-burnin.sh
|
|
#
|
|
################################################################################
|
|
|
|
################################################################################
|
|
# PRE-EXECUTION VALIDATION
|
|
################################################################################
|
|
|
|
# Check if running as root
|
|
if [ "$(id -u)" -ne 0 ]; then
|
|
echo "ERROR: Must be run as root" >&2
|
|
exit 2
|
|
fi
|
|
|
|
# Check required dependencies
|
|
readonly DEPENDENCIES="awk badblocks grep sed sleep smartctl"
|
|
for dependency in ${DEPENDENCIES}; do
|
|
if ! command -v "${dependency}" > /dev/null 2>&1; then
|
|
echo "ERROR: Command '${dependency}' not found" >&2
|
|
exit 2
|
|
fi
|
|
done
|
|
|
|
readonly USAGE=\
|
|
"NAME
|
|
$(basename "$0") -- disk burn-in program
|
|
|
|
SYNOPSIS
|
|
$(basename "$0") [-h] [-e] [-f] [-o <directory>] [-x] <disk>
|
|
|
|
DESCRIPTION
|
|
A script to simplify the process of burning-in disks. Only intended for use
|
|
on disks which do not contain any data, such as new disks or disks which
|
|
are being tested or re-purposed.
|
|
|
|
The script runs in dry-run mode by default, so you can check the sleep
|
|
durations and to insure that the sequence of commands suits your needs. In
|
|
dry-run mode the script does not actually perform any SMART tests or invoke
|
|
the sleep or badblocks programs.
|
|
|
|
In order to perform tests on drives, you will need to provide the -f option.
|
|
|
|
OPTIONS
|
|
-h Show help text
|
|
-e Show extended help text
|
|
-f Force script to run in destructive mode
|
|
ALL DATA ON THE DISK WILL BE LOST!
|
|
-o <directory> Write log files to <directory> (default: $(pwd))
|
|
-x Run full pass of badblocks instead of exiting on first error
|
|
<disk> Disk to burn-in (/dev/ may be omitted)
|
|
|
|
EXAMPLES
|
|
$(basename "$0") sda
|
|
run in dry-run mode on disk /dev/sda
|
|
|
|
$(basename "$0") -f /dev/sdb
|
|
run in destructive, non-dry mode on disk /dev/sdb
|
|
|
|
$(basename "$0") -f -o ~/burn-in-logs sdc
|
|
run in destructive, non-dry mode on disk /dev/sdc and
|
|
write the log files to ~/burn-in-logs directory
|
|
"
|
|
readonly USAGE_2=\
|
|
"EXIT STATUS
|
|
exit 0: script finishes successfully
|
|
exit 2: dependencies are missing
|
|
not running as 'root'
|
|
illegal options are provided
|
|
|
|
NOTES
|
|
Be warned that:
|
|
|
|
1> The script runs badblocks in destructive mode, which erases any data
|
|
on the disk.
|
|
|
|
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
|
|
!!!! ALL DATA ON THE DISK WILL BE LOST! BE CAREFUL! !!!!
|
|
!!!! DO NOT RUN THIS SCRIPT ON DISKS CONTAINING VALUABLE DATA !!!!
|
|
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
|
|
|
|
2> Run times for large disks can be several days. Use tmux or screen
|
|
to test multiple disks in parallel.
|
|
|
|
3> Must be run as 'root'.
|
|
|
|
4> The script has the following dependencies:
|
|
|
|
smartmontools, available at https://www.smartmontools.org
|
|
Uses: grep, awk, sed, sleep, badblocks
|
|
|
|
Performs this test sequence:
|
|
|
|
1> Run SMART short test
|
|
2> Run badblocks
|
|
3> Run SMART extended test
|
|
|
|
The script sleeps after starting each SMART test, using a duration based on
|
|
the polling interval reported by the disk, after which the script will poll
|
|
the disk to verify the self-test has completed.
|
|
|
|
Full SMART information is pulled after each SMART test. All output except
|
|
for the sleep command is written to both stdout and the log.
|
|
|
|
You should monitor the burn-in progress and watch for errors, particularly
|
|
any errors reported by badblocks, or these SMART errors:
|
|
|
|
5 Reallocated_Sector_Ct
|
|
196 Reallocated_Event_Count
|
|
197 Current_Pending_Sector
|
|
198 Offline_Uncorrectable
|
|
|
|
These indicate possible problems with the drive. You therefore may wish to
|
|
abort the remaining tests and proceed with an RMA exchange for new drives or
|
|
discard old ones. Please note that this list is not exhaustive.
|
|
|
|
The script extracts the drive model and serial number and forms a log file-
|
|
name of the form 'burnin-[model]_[serial number].log'.
|
|
|
|
badblocks is invoked with a block size of 4096, the -wsv options, and the
|
|
-o option to instruct it to write the list of bad blocks found (if any) to
|
|
a file named 'burnin-[model]_[serial number].bb'.
|
|
|
|
Before using the script on FreeBSD systems (including FreeNAS) you must
|
|
first execute this sysctl command to alter the kernel's geometry debug
|
|
flags. This allows badblocks to write to the entire disk:
|
|
|
|
sysctl kern.geom.debugflags=0x10
|
|
|
|
Also note that badblocks may issue the following warning under FreeBSD /
|
|
FreeNAS, which can safely be ignored as it has no effect on testing:
|
|
|
|
set_o_direct: Inappropiate ioctl for device
|
|
|
|
Tested operating systems:
|
|
|
|
FreeNAS 9.10.2 (FreeBSD 10.3-STABLE)
|
|
FreeNAS 11.1-U7 (FreeBSD 11.1-STABLE)
|
|
FreeNAS 11.2-U8 (FreeBSD 11.2-STABLE)
|
|
Ubuntu Server 16.04.2 LTS
|
|
CentOS 7.0
|
|
Tiny Core Linux 11.1
|
|
|
|
Tested disks:
|
|
|
|
Intel
|
|
DC S3700 SSD
|
|
Model 320 Series SSD
|
|
HGST
|
|
Deskstar NAS (HDN724040ALE640)
|
|
Ultrastar 7K4000 (HUS724020ALE640)
|
|
Ultrastar He10
|
|
Ultrastar He12
|
|
Western Digital
|
|
Black (WD6001FZWX)
|
|
Gold
|
|
Re (WD4000FYYZ)
|
|
Seagate
|
|
IronWolf NAS HDD 12TB (ST12000VN0008)
|
|
|
|
VERSIONS
|
|
Written by Keith Nash, March 2017
|
|
|
|
KN, 8 Apr 2017:
|
|
Added minimum test durations because some devices don't return accurate
|
|
values.
|
|
Added code to clean up the log file, removing copyright notices, etc.
|
|
No longer echo 'smartctl -t' output to log file as it imparts no useful
|
|
information.
|
|
Emit test results after tests instead of full 'smartctl -a' output.
|
|
Emit full 'smartctl -x' output at the end of all testing.
|
|
Minor changes to log output and formatting.
|
|
|
|
KN, 12 May 2017:
|
|
Added code to poll the disk and check for completed self-tests.
|
|
|
|
As noted above, some disks don't report accurate values for the short
|
|
and extended self-test intervals, sometimes by a significant amount.
|
|
The original approach using 'fudge' factors wasn't reliable and the
|
|
script would finish even though the SMART self-tests had not completed.
|
|
The new polling code helps insure that this doesn't happen.
|
|
|
|
Fixed code to work around annoying differences between sed's behavior
|
|
on Linux and FreeBSD.
|
|
|
|
KN, 8 Jun 2017
|
|
Modified parsing of short and extended test durations to accommodate the
|
|
values returned by larger drives; we needed to strip out the '(' and ')'
|
|
characters surrounding the integer value in order to fetch it reliably.
|
|
|
|
KN, 19 Aug 2020
|
|
Changed DRY_RUN value so that dry runs are no longer the default
|
|
setting.
|
|
Changed badblocks call to exit immediately on first error.
|
|
Set logging directory to current working directory using pwd command.
|
|
Reduced default tests so that we run:
|
|
1> Short SMART test
|
|
2> badblocks
|
|
3> Extended SMART test
|
|
|
|
MS, 9 Sep 2020
|
|
Add .editorconfig to streamlime editor behavior for developers.
|
|
Remove dependencies on pcregrep and tr.
|
|
Add documentation to functions and complex statements.
|
|
Reduce code duplication, simplify and decouple code where possible.
|
|
Improve portability and resiliency.
|
|
Check availability of dependencies during runtim.
|
|
Check for root privileges during runtime.
|
|
Add option parsing, most notably (-h)elp and -f for non-dry-run mode.
|
|
Add dry_run_wrapper() function.
|
|
Add disk type detection to skip badblocks for non-mechanical drives.
|
|
|
|
KN, 5 Oct 2020
|
|
Added -x option to control the badblocks -e option, allowing extended testing.
|
|
Added smartctl to the list of dependencies.
|
|
Changed disk type detection so that we assume all drives are mechanical drives
|
|
unless they explicitly return 'Solid State Drive' for Rotational Rate.
|
|
Removed datestamp from every line of log output, only emitting it in log headers.
|
|
Minor reformatting."
|
|
|
|
# badblocks default -e option is 1, stop testing if a single error occurs
|
|
BB_E_ARG=1
|
|
|
|
# parse options
|
|
while getopts ':hefo:x' option; do
|
|
case "${option}" in
|
|
h) echo "${USAGE}"
|
|
exit
|
|
;;
|
|
e) echo "${USAGE}"
|
|
echo "${USAGE_2}"
|
|
exit
|
|
;;
|
|
f) readonly DRY_RUN=0
|
|
;;
|
|
o) LOG_DIR="${OPTARG}"
|
|
;;
|
|
x) BB_E_ARG=0
|
|
;;
|
|
:) printf 'Missing argument for -%s\n' "${OPTARG}" >&2
|
|
echo "${USAGE}" >&2
|
|
exit 2
|
|
;;
|
|
\?) printf 'Illegal option: -%s\n' "${OPTARG}" >&2
|
|
echo "${USAGE}" >&2
|
|
exit 2
|
|
;;
|
|
esac
|
|
done
|
|
shift $(( OPTIND - 1 ))
|
|
|
|
if [ -z "$1" ]; then
|
|
echo "ERROR: Missing disk argument" >&2
|
|
echo "${USAGE}" >&2
|
|
exit 2
|
|
fi
|
|
|
|
################################################################################
|
|
# CONSTANTS
|
|
################################################################################
|
|
|
|
readonly BB_E_ARG
|
|
|
|
# Drive to burn-in
|
|
DRIVE="$1"
|
|
# prepend /dev/ if necessary
|
|
if ! printf '%s' "${DRIVE}" | grep "/dev/\w*" > /dev/null 2>&1; then
|
|
DRIVE="/dev/${DRIVE}"
|
|
fi
|
|
readonly DRIVE
|
|
|
|
if [ ! -e "$DRIVE" ]; then
|
|
echo "ERROR: Disk does not exist: $DRIVE" >&2
|
|
echo "${USAGE}" >&2
|
|
exit 2
|
|
fi
|
|
|
|
# Set to working directory if -o <directory> wasn't provided
|
|
[ -z "${LOG_DIR}" ] && LOG_DIR="$(pwd)"
|
|
# Trim trailing slashes
|
|
LOG_DIR="$(printf '%s' "${LOG_DIR}" | awk '{gsub(/\/+$/, ""); printf $1}')"
|
|
readonly LOG_DIR
|
|
|
|
# System information
|
|
readonly HOSTNAME="$(hostname)"
|
|
readonly OS_FLAVOR="$(uname)"
|
|
|
|
# SMART static information
|
|
readonly SMART_INFO="$(smartctl --info "${DRIVE}")"
|
|
readonly SMART_CAPABILITIES="$(smartctl --capabilities "${DRIVE}")"
|
|
|
|
##################################################
|
|
# Get SMART information value.
|
|
# Globals:
|
|
# SMART_INFO
|
|
# Arguments:
|
|
# value identifier:
|
|
# !!! Only TWO WORD indentifiers are supported !!!
|
|
# !!! Querying e.g. "ATA Version is" will fail !!!
|
|
# - Device Model
|
|
# - Model Family
|
|
# - Serial Number
|
|
# Outputs:
|
|
# Write value to stdout.
|
|
##################################################
|
|
get_smart_info_value() {
|
|
# $1=$2=""; select all but first two columns
|
|
# gsub(/^[ \t]+|[ \t]+$/, ""); replace leading and trailing whitespace
|
|
# gsub(/ /, "_"); replace remaining spaces with underscores
|
|
# printf $1 print result without newline at the end
|
|
printf '%s' "${SMART_INFO}" \
|
|
| grep "$1" \
|
|
| awk '{$1=$2=""; gsub(/^[ \t]+|[ \t]+$/, ""); gsub(/ /, "_"); printf $1}'
|
|
}
|
|
|
|
##################################################
|
|
# Get SMART recommended test duration, in minutes.
|
|
# Globals:
|
|
# SMART_CAPABILITIES
|
|
# Arguments:
|
|
# test type:
|
|
# - Short
|
|
# - Extended
|
|
# - Conveyance
|
|
# Outputs:
|
|
# Write duration to stdout.
|
|
##################################################
|
|
get_smart_test_duration() {
|
|
# '/'"$1"' self-test routine/ match duration depending on test type arg
|
|
# getline; jump to next line
|
|
# gsub(/\(|\)/, ""); remove parantheses
|
|
# printf $4 print 4th column without newline at the end
|
|
printf '%s' "${SMART_CAPABILITIES}" \
|
|
| awk '/'"$1"' self-test routine/{getline; gsub(/\(|\)/, ""); printf $4}'
|
|
}
|
|
|
|
# Get disk model
|
|
DISK_MODEL="$(get_smart_info_value "Device Model")"
|
|
[ -z "${DISK_MODEL}" ] && DISK_MODEL="$(get_smart_info_value "Model Family")"
|
|
[ -z "${DISK_MODEL}" ] && DISK_MODEL="$(get_smart_info_value "Model Number")"
|
|
readonly DISK_MODEL
|
|
|
|
# Get disk type; unless we get 'Solid State Device' as return value, assume
|
|
# we have a mechanical drive.
|
|
DISK_TYPE="$(get_smart_info_value "Rotation Rate")"
|
|
if printf '%s' "${DISK_TYPE}" | grep -i "solid_state_device" > /dev/null 2>&1; then
|
|
DISK_TYPE="SSD"
|
|
fi
|
|
readonly DISK_TYPE
|
|
|
|
# Get disk serial number
|
|
readonly SERIAL_NUMBER="$(get_smart_info_value "Serial Number")"
|
|
|
|
# SMART short test duration
|
|
readonly SHORT_TEST_MINUTES="$(get_smart_test_duration "Short")"
|
|
readonly SHORT_TEST_SECONDS="$(( SHORT_TEST_MINUTES * 60))"
|
|
|
|
# SMART extended test duration
|
|
readonly EXTENDED_TEST_MINUTES="$(get_smart_test_duration "Extended")"
|
|
readonly EXTENDED_TEST_SECONDS="$(( EXTENDED_TEST_MINUTES * 60 ))"
|
|
|
|
# Maximum duration the completion status is polled
|
|
readonly POLL_TIMEOUT_HOURS=4
|
|
readonly POLL_TIMEOUT_SECONDS="$(( POLL_TIMEOUT_HOURS * 60 * 60))"
|
|
|
|
# Sleep interval between completion status polls
|
|
readonly POLL_INTERVAL_SECONDS=15
|
|
|
|
# Form log file names
|
|
readonly LOG_FILE="${LOG_DIR}/burnin-${DISK_MODEL}_${SERIAL_NUMBER}.log"
|
|
readonly BB_File="${LOG_DIR}/burnin-${DISK_MODEL}_${SERIAL_NUMBER}.bb"
|
|
|
|
################################################################################
|
|
# FUNCTIONS
|
|
################################################################################
|
|
|
|
##################################################
|
|
# Log informational message.
|
|
# Globals:
|
|
# LOG_FILE
|
|
# Arguments:
|
|
# Message to log.
|
|
# Outputs:
|
|
# Write message to stdout and log file.
|
|
##################################################
|
|
log_info() {
|
|
# now="$(date +"%F %T %Z")"
|
|
# printf "%s\n" "[${now}] $1" | tee -a "${LOG_FILE}"
|
|
printf "%s\n" "$1" | tee -a "${LOG_FILE}"
|
|
}
|
|
|
|
##################################################
|
|
# Log emphasized header message.
|
|
# Arguments:
|
|
# Message to log.
|
|
##################################################
|
|
log_header() {
|
|
log_info "+-----------------------------------------------------------------------------"
|
|
log_info "+ $1: $(date)"
|
|
log_info "+-----------------------------------------------------------------------------"
|
|
}
|
|
|
|
##################################################
|
|
# Ensure log directory exists and remove old logs.
|
|
# Globals:
|
|
# LOG_DIR
|
|
# LOG_FILE
|
|
# Arguments:
|
|
# None
|
|
##################################################
|
|
init_log() {
|
|
mkdir -p -- "${LOG_DIR}" || exit 2
|
|
[ -e "${LOG_FILE}" ] && rm -- "${LOG_FILE}"
|
|
}
|
|
|
|
##################################################
|
|
# Remove redundant messages from log.
|
|
# Globals:
|
|
# LOG_FILE
|
|
# OS_FLAVOR
|
|
# Arguments:
|
|
# None
|
|
##################################################
|
|
cleanup_log() {
|
|
if [ "${OS_FLAVOR}" = "Linux" ]; then
|
|
sed -i -e '/Copyright/d' "${LOG_FILE}"
|
|
sed -i -e '/=== START OF READ/d' "${LOG_FILE}"
|
|
sed -i -e '/SMART Attributes Data/d' "${LOG_FILE}"
|
|
sed -i -e '/Vendor Specific SMART/d' "${LOG_FILE}"
|
|
sed -i -e '/SMART Error Log Version/d' "${LOG_FILE}"
|
|
fi
|
|
|
|
if [ "${OS_FLAVOR}" = "FreeBSD" ]; then
|
|
sed -i '' -e '/Copyright/d' "${LOG_FILE}"
|
|
sed -i '' -e '/=== START OF READ/d' "${LOG_FILE}"
|
|
sed -i '' -e '/SMART Attributes Data/d' "${LOG_FILE}"
|
|
sed -i '' -e '/Vendor Specific SMART/d' "${LOG_FILE}"
|
|
sed -i '' -e '/SMART Error Log Version/d' "${LOG_FILE}"
|
|
fi
|
|
}
|
|
|
|
##################################################
|
|
# Log command in dry-run mode, run command otherwise.
|
|
# Globals:
|
|
# DRY_RUN
|
|
# Arguments:
|
|
# Command to run.
|
|
##################################################
|
|
dry_run_wrapper() {
|
|
if [ -z "$DRY_RUN" ]; then
|
|
log_info "DRY RUN: $*"
|
|
return 0
|
|
fi
|
|
eval "$@"
|
|
}
|
|
|
|
##################################################
|
|
# Log runtime information about current burn-in.
|
|
# Globals:
|
|
# HOSTNAME
|
|
# OS_FLAVOR
|
|
# DRIVE
|
|
# DISK_TYPE
|
|
# DISK_MODEL
|
|
# SERIAL_NUMBER
|
|
# SHORT_TEST_MINUTES
|
|
# SHORT_TEST_SECONDS
|
|
# EXTENDED_TEST_MINUTES
|
|
# EXTENDED_TEST_SECONDS
|
|
# LOG_FILE
|
|
# BB_File
|
|
# Arguments:
|
|
# None
|
|
##################################################
|
|
log_runtime_info() {
|
|
log_info "Host: ${HOSTNAME}"
|
|
log_info "OS: ${OS_FLAVOR}"
|
|
log_info "Drive: ${DRIVE}"
|
|
log_info "Disk Type: ${DISK_TYPE}"
|
|
log_info "Drive Model: ${DISK_MODEL}"
|
|
log_info "Serial Number: ${SERIAL_NUMBER}"
|
|
log_info "Short test duration: ${SHORT_TEST_MINUTES} minutes"
|
|
log_info " ${SHORT_TEST_SECONDS} seconds"
|
|
log_info "Extended test duration: ${EXTENDED_TEST_MINUTES} minutes"
|
|
log_info " ${EXTENDED_TEST_SECONDS} seconds"
|
|
log_info "Log file: ${LOG_FILE}"
|
|
log_info "Bad blocks file: ${BB_File}"
|
|
}
|
|
|
|
##################################################
|
|
# Poll repeatedly whether SMART self-test has completed.
|
|
# Globals:
|
|
# DRIVE
|
|
# POLL_INTERVAL_SECONDS
|
|
# POLL_TIMEOUT_SECONDS
|
|
# Arguments:
|
|
# None
|
|
# Returns:
|
|
# 0 if success or failure.
|
|
# 1 if timeout threshold exceeded.
|
|
##################################################
|
|
poll_selftest_complete() {
|
|
l_poll_duration_seconds=0
|
|
while [ "${l_poll_duration_seconds}" -lt "${POLL_TIMEOUT_SECONDS}" ]; do
|
|
smartctl --all "${DRIVE}" \
|
|
| grep -i "The previous self-test routine completed" > /dev/null 2>&1
|
|
l_status="$?"
|
|
if [ "${l_status}" -eq 0 ]; then
|
|
log_info "SMART self-test succeeded"
|
|
return 0
|
|
fi
|
|
smartctl --all "${DRIVE}" \
|
|
| grep -i "of the test failed\." > /dev/null 2>&1
|
|
l_status="$?"
|
|
if [ "${l_status}" -eq 0 ]; then
|
|
log_info "SMART self-test failed"
|
|
return 0
|
|
fi
|
|
sleep "${POLL_INTERVAL_SECONDS}"
|
|
l_poll_duration_seconds="$(( l_poll_duration_seconds + POLL_INTERVAL_SECONDS ))"
|
|
done
|
|
log_info "SMART self-test timeout threshold exceeded"
|
|
return 1
|
|
}
|
|
|
|
##################################################
|
|
# Run SMART test and log results.
|
|
# Globals:
|
|
# DRIVE
|
|
# LOG_FILE
|
|
# Arguments:
|
|
# Test type:
|
|
# - short
|
|
# - long
|
|
# Test duration in seconds.
|
|
##################################################
|
|
run_smart_test() {
|
|
log_header "Running SMART $1 test"
|
|
dry_run_wrapper "smartctl --test=\"$1\" \"${DRIVE}\""
|
|
log_info "SMART $1 test started, awaiting completion for $2 seconds ..."
|
|
dry_run_wrapper "sleep \"$2\""
|
|
dry_run_wrapper "poll_selftest_complete"
|
|
dry_run_wrapper "smartctl --log=selftest \"${DRIVE}\" | tee -a \"${LOG_FILE}\""
|
|
log_info "Finished SMART $1 test"
|
|
}
|
|
|
|
##################################################
|
|
# Run badblocks test.
|
|
# !!! ALL DATA ON THE DISK WILL BE LOST !!!
|
|
# Globals:
|
|
# BB_File
|
|
# DISK_TYPE
|
|
# DRIVE
|
|
# Arguments:
|
|
# None
|
|
##################################################
|
|
run_badblocks_test() {
|
|
log_header "Running badblocks test"
|
|
if [ "${DISK_TYPE}" != "SSD" ]; then
|
|
dry_run_wrapper "badblocks -b 4096 -wsv -e ${BB_E_ARG} -o \"${BB_File}\" \"${DRIVE}\""
|
|
else
|
|
log_info "SKIPPED: badblocks for ${DISK_TYPE} device"
|
|
fi
|
|
log_info "Finished badblocks test"
|
|
}
|
|
|
|
##################################################
|
|
# Log extensive SMART and non-SMART drive information.
|
|
# Globals:
|
|
# DRIVE
|
|
# LOG_FILE
|
|
# Arguments:
|
|
# None
|
|
##################################################
|
|
log_full_device_info() {
|
|
log_header "Drive information"
|
|
dry_run_wrapper "smartctl --xall --vendorattribute=7,hex48 \"${DRIVE}\" | tee -a \"${LOG_FILE}\""
|
|
}
|
|
|
|
##################################################
|
|
# Main function of script.
|
|
# Globals:
|
|
# SHORT_TEST_SECONDS
|
|
# EXTENDED_TEST_SECONDS
|
|
# Arguments:
|
|
# None
|
|
##################################################
|
|
main() {
|
|
init_log
|
|
log_header "Started burn-in"
|
|
|
|
log_runtime_info
|
|
|
|
# test sequence
|
|
run_smart_test "short" "${SHORT_TEST_SECONDS}"
|
|
run_badblocks_test
|
|
run_smart_test "long" "${EXTENDED_TEST_SECONDS}"
|
|
|
|
log_full_device_info
|
|
|
|
log_header "Finished burn-in"
|
|
cleanup_log
|
|
}
|
|
|
|
# Entrypoint
|
|
main
|