สิ่งนี้ไม่ต้องการสิทธิ์รูท (แต่คำสั่งเหล่านี้จำนวนมากใช้และขึ้นอยู่กับ bashisms ดังนั้นคำสั่งเหล่านี้จะไม่ทำงานในเชลล์ที่สอดคล้องกับ POSIX ทั้งหมด):
มีวิธีที่รวดเร็วในการถามเกี่ยวกับ:sdX
grep -H . /sys/block/sda/{capability,uevent,removable,device/{model,type,vendor,uevent}}
/sys/block/sda/capability:52
/sys/block/sda/uevent:MAJOR=8
/sys/block/sda/uevent:MINOR=0
/sys/block/sda/uevent:DEVNAME=sda
/sys/block/sda/uevent:DEVTYPE=disk
/sys/block/sda/removable:0
/sys/block/sda/device/model:WDC WD360GD-00FN
/sys/block/sda/device/type:0
/sys/block/sda/device/vendor:ATA
/sys/block/sda/device/uevent:DEVTYPE=scsi_device
/sys/block/sda/device/uevent:DRIVER=sd
/sys/block/sda/device/uevent:MODALIAS=scsi:t-0x00
จริงๆcapability
ไฟล์ที่น่าสนใจคือ ใน Debian ของฉันฉันมีgenhd.h
ไฟล์ดังนั้น:
eval $(sed -ne '
s/#define.*GENHD_FL_\([A-Z0-9_]*\)[ \t]*\([0-9]*\) \?.*$/GENHD_FLAGS[\2]="\1"/p
' /usr/src/linux-headers-2.6.32-5-common-openvz/include/linux/genhd.h)
diskCapa=$(</sys/block/sda/capability)
for i in ${!GENHD_FLAGS[@]};do
(( diskCapa & i )) && echo ${GENHD_FLAGS[i]}
done
MEDIA_CHANGE_NOTIFY
UP
SUPPRESS_PARTITION_INFO
diskCapa=$(</sys/block/sdd/capability)
for i in ${!GENHD_FLAGS[@]};do
(( diskCapa & i )) && echo ${GENHD_FLAGS[i]}
done
REMOVABLE
MEDIA_CHANGE_NOTIFY
UP
SUPPRESS_PARTITION_INFO
เลยเพียงแค่รู้ว่าตั้งค่าสถานะแล้วremovable
:
grep REMOVABL /usr/src/linux-headers-3.2.0-4-common/include/linux/genhd.h
#define GENHD_FL_REMOVABLE 1
ดังนั้น
for disk in sd{a,b,c,d,e,f,g,h} ; do
(( $(< /sys/block/$disk/capability ) & 1 )) && echo $disk is removable
done
ทำงานโดยการทดสอบว่าcapability
ค่า (ซึ่งเป็น 52 ในsda
ตัวอย่างของฉันด้านบน) มีการ1
ตั้งค่าบิต (เช่นไม่ว่าจะเป็นจำนวนคี่)
แต่ลีนุกซ์ให้ค่าสถานะทั้งหมด/sys
ดังนั้นการถามหา/sys/block/sdX/removable
จึงง่ายกว่ามาก! ;-)
ดังนั้นคีย์ USB อาจเป็นได้removable
แต่เนื่องจากมีอุปกรณ์พกพาจำนวนมากฉันต้องการให้แน่ใจว่าขนาดของสื่อนั้นมากกว่า 0 (เช่นถาด CD-ROM ที่ไม่ได้โหลดสำหรับตัวอย่าง)
และว่าอุปกรณ์เป็นไม่ได้ใน ใช้: ในการดูที่sdX/trace/enable
ไม่ได้ผูก:
หมายเหตุ: ทั้งหมดนี้ทดสอบอย่างดีกับ bash v4.2 +
ภายใต้ ทุบตีคุณสามารถใช้วิธีนี้อย่างรวดเร็วและมีประสิทธิภาพ:
for disk in /sys/block/* ; do
[ -f "$disk/removable" ] && [ $(<"$disk/removable") -gt 0 ] &&
[ -f "$disk/size" ] && [ $(<"$disk/size") -gt 0 ] &&
[ -f "$disk/trace/enable" ] && [ -z "$(<"$disk/trace/enable")" ] &&
echo "${disk##*/} $(($(<"$disk/size")/1953125))G $(<"$disk/device/model")"
done
ในระบบของฉันมีปุ่ม USB 4 อัน แต่หนึ่งในนั้นsde
ก็ติดตั้งอยู่แล้วดังนั้นคำสั่งก่อนหน้านี้:
sdd 8G Trans-It Drive
sdf 7G Storage Media
sdg 4G silicon-power
สคริปต์ของฉัน:
มีการทุบตีเล็กน้อยฟังก์ชั่นฉันเขียนเพื่อติดตั้ง Debian Live ที่อัปเกรดแล้ว
#!/bin/bash
txtsize() {
local _c=$1 _i=0 _a=(b K M G T P)
while [ ${#_c} -gt 3 ] ; do
((_i++))
_c=$((_c>>10))
done
_c=000$(( ( $1*1000 ) >> ( 10*_i ) ))
((_i+=${3:-0}))
printf -v ${2:-REPLY} "%.2f%s" ${_c:0:${#_c}-3}.${_c:${#_c}-3} ${_a[_i]}
}
# ส่วนแรกแสดงผลขนาดที่มนุษย์อ่านได้เท่านั้น ฟังก์ชั่นเริ่มต้นที่นั่น
chooseFreeUsbKey() {
local _lUdisk _lUsize _lUdialog=dialog # whiptail # gdialog
local -A _lUdevices
unset ${1:-REPLY}
for _lUdisk in /sys/block/*; do
[ -f $_lUdisk/removable ] && [ $(<$_lUdisk/removable) -gt 0 ] &&
[ -f $_lUdisk/size ] && [ $(<$_lUdisk/size) -gt 0 ] &&
txtsize $(<$_lUdisk/size)*512 _lUsize &&
[ -f $_lUdisk/trace/enable ] && [ -z "$(<$_lUdisk/trace/enable)" ] &&
_lUdevices[${_lUdisk##*/}]="$_lUsize $(<$_lUdisk/device/model)"
done
case ${#_lUdevices[@]} in
0 ) ;; # echo Sorry no key found. ;;
1 ) IFS=§ read -a ${1:-REPLY} \
<<< "${!_lUdevices[@]}§${_lUdevices[@]%% *}§${_lUdevices[@]#* }";;
* ) declare -a menu
for _lUdisk in ${!_lUdevices[@]}; do
menu+=($_lUdisk "${_lUdevices[$_lUdisk]}")
done
_lUdisk=$($_lUdialog --menu "Choose a USB stick" \
$((LINES-3)) $((COLUMNS-3)) $((LINES-8)) \
"${menu[@]}" 2>&1 >/dev/tty)
IFS=§ read -a ${1:-REPLY} \
<<< "$_lUdisk§${_lUdevices[$_lUdisk]%% *}§${_lUdevices[$_lUdisk]#* }"
esac
}
สิ่งนี้จะกำหนดคำตอบในฐานะอาร์เรย์ให้กับตัวแปรที่กำหนดเป็นอาร์กิวเมนต์แรกหรือตัวแปร$REPLY
:
chooseFreeUsbKey stick
echo "$stick"
sdf
echo "${stick[1]}"
7.26G
echo "${stick[2]}"
Storage Media
(เขตข้อมูลสุดท้ายอาจมีช่องว่าง)