X-Git-Url: http://git.grml.org/?p=grml-hwinfo.git;a=blobdiff_plain;f=grml-hwinfo;h=a2203bc834a24743d75d9832601bba1e4c2545a4;hp=158e75074ffd2be83f97c1873ebb15d368f4ef0c;hb=HEAD;hpb=f15244e3b559888efd1e5c29f38db2eb79fc22a2 diff --git a/grml-hwinfo b/grml-hwinfo index 158e750..97ad572 100755 --- a/grml-hwinfo +++ b/grml-hwinfo @@ -1,10 +1,9 @@ -#!/bin/sh +#!/bin/bash # Filename: grml-hwinfo # Purpose: get hardware information # Authors: grml-team (grml.org), (c) Michael Prokop # Bug-Reports: see http://grml.org/bugs/ # License: This file is licensed under the GPL v2. -# Latest change: Mon Aug 27 18:32:27 CEST 2007 [mika] ################################################################################ # Notice: Some ideas have been taken from # http://club.black.co.at/david/hwdb/infodump @@ -13,195 +12,496 @@ # variables UNAME="$(uname -r)" -PN="$(basename $0)" -VERSION='0.3' +PN="$(basename "$0")" +[ -n "$WORKING_DIR" -a -d "$WORKING_DIR" ] || WORKING_DIR=$(pwd) +VERSION='***UNRELEASED***' -[ -n "$INFOFILE" ] || INFOFILE='info.tar.bz2' +# data collection should not be affected by user locale +export LANG=C +export LC_ALL=C +TIMESTAMP='+%F--%H-%M-%S-%Z' +DATE="$(date $TIMESTAMP)" -if [ "$(id -u)" != 0 ] ; then - NOTROOT=1 - echo "Notice: you are running $PN without root permissions. Not all information will be collected." - echo +# defaults +GENERATE_FILE='1' +GENERATE_DIRECTORY='' +_opt_output_directory=false +_opt_output_file=false +_opt_quiet=false +_opt_force=false + +usage() { + echo " + This tool collects information of the hardware it is being executed on. + It can be executed as normal user to collect some basic information or + (recommended) with root permissions to collect more system information. + If executed without any options a file named grml-hwinfo-TIMESTAMP.tar.bz2 + storing all collected information is created in the current working directory. + + Options: + + -b, --both Create directory + file grml-hwinfo-TIMESTAMP.tar.bz2 + -d, --directory Create grml-hwinfo-TIMESTAMP as a directory (no file) + -f, --file Create grml-hwinfo-TIMESTAMP.tar.bz2 [default action] + -h, --help Display this help message + -q, --quiet Don't display informational text (useful for cron usage) + --force Don't abort but overwrite possibly existing output file + --output-directory Store output files in specified directory + --output-file Store output in specified filename (tar.XX format) + " +} + +CMDLINE_OPTS=output-directory:,output-file:,both,directory,file,help,quiet,force +_opt_temp=$(getopt --name grml-hwinfo -o +bdfhq --long $CMDLINE_OPTS -- "$@") +if [ $? -ne 0 ]; then + echo "Try 'grml-hwinfo --help' for more information." >&2 + exit 1 fi +eval set -- "$_opt_temp" -# check whether a binary is available and executable -exectest() { -if [ -z "$1" ] ; then - echo 'Usage: exectest '>&2 - return 1 -else - test -e "$(which $1)" && return 0 || return 1 +while :; do + case "$1" in + --help|-h) + usage ; exit 0 + ;; + --output-directory) + shift; OUTDIRNAME="$1" + GENERATE_DIRECTORY='1' + _opt_output_directory=true + $_opt_output_file && GENERATE_FILE='1' || GENERATE_FILE='' + ;; + --output-file) + shift; OUTFILE="$1" + GENERATE_FILE='1' + _opt_output_file=true + $_opt_output_directory && GENERATE_DIRECTORY='1' || GENERATE_DIRECTORY='' + ;; + -d|--directory) + GENERATE_DIRECTORY='1' + GENERATE_FILE='' + ;; + -f|--file) + GENERATE_DIRECTORY='' + GENERATE_FILE='1' + ;; + -b|--both) + GENERATE_DIRECTORY='1' + GENERATE_FILE='1' + ;; + -q|--quiet) + _opt_quiet=true + ;; + --force) + _opt_force=true + ;; + --) + shift; break + ;; + *) + echo "Internal getopt error!" >&2 + exit 1 + ;; + esac + shift +done + +if ! $_opt_quiet ; then + echo "$PN ${VERSION} - collect hardware information" fi -} -# echo a list of all disks and their size -# taken from http://cvs.debian.org/fai/lib/disk-info -diskandsize() { - local isdisk major minor blocks device suffix - while read major minor blocks device suffix; do - isdisk=1 - # skip ide cdrom - [ -f /proc/ide/$device/media ] && grep -q cdrom /proc/ide/$device/media && isdisk=0 - [ "$isdisk" -eq 1 ] && echo "$device $blocks" - done -} +# Generate output/temporary directory name & path, and output file path +[ -n "$OUTDIRNAME" ] || OUTDIRNAME="grml-hwinfo-${DATE}" +if $_opt_output_directory ; then + OUTDIR="${OUTDIRNAME}" +else + OUTDIR="${WORKING_DIR}/${OUTDIRNAME}" +fi + +if $_opt_force ; then + mkdir -p "${OUTDIR}" +else + mkdir "${OUTDIR}" || { echo "Directory '${OUTDIR}' already exists, aborting." >&2 ; exit 1; } +fi + +if [ -n "$GENERATE_FILE" ] ; then + [ -n "$OUTFILE" ] && OUTFILE_="$OUTFILE" || OUTFILE_="${OUTDIR}.tar.bz2" + if ! $_opt_force ; then + [ -e "${OUTFILE_}" ] && { echo "File '${OUTFILE_}' already exists, aborting." >&2 ; rm -r "${OUTDIR}"; exit 1; } + fi + OUTFILE=${OUTFILE_} + touch "${OUTFILE}" +fi + +if [ "$(id -u)" != "0" ] ; then + NOTROOT=1 + $_opt_quiet || echo "W: Running without root permissions. Not all data will be collected." +fi -list_disks() { -# print only every second entry; used by disk_info -i=0 -for ent in $@; do - if [ "$i" -eq 0 ]; then - echo $ent - i=1 +# check whether a binary is available and executable +exectest() { + if [ -z "$1" ] ; then + echo 'Usage: exectest '>&2 + return 1 + else + if test -e "$(which "$1")" ; then + return 0 else - i=0 + if ! grep -q "^$1"'$' missing_tools 2>/dev/null ; then + $_opt_quiet || echo "$1" >> missing_tools + fi + return 1 fi -done + fi } disk_info() { - # the variable holds a space separated list of devices and their block size - device_size=`grep -E ' cciss/c.d.$| ida/c.d.$| rd/c.d.$| hd.$| sd.$|/disc$' /proc/partitions | diskandsize` - # a list of all local disks, without size - disklist=`list_disks $device_size` + # the variable holds a newline separated list of disk block devices, excluding loopback and CD-ROM devices + disklist=$(lsblk -nd -o NAME -e 7,11) } -echo "Running grml-hwinfo ${VERSION} - collecting hardware information." -echo "Please send feedback, bugreports, feature requests to Michael Prokop ." -echo "" +# return list of all network devices in array "${niclist[@]}" +get_network_devices() { + local interface + niclist=() + for interface in /sys/class/net/* ; do + [ -e "${interface}" ] || continue + interface=$(basename "${interface}") + # [ "${interface}" = "lo" ] && continue + niclist+=("${interface}") + done +} -mkdir info || exit -1 -( - cd info +# Check if X server is running +# +# If xset is missing, we rely on the existence of the $DISPLAY variable. +NO_DISPLAY=0 +if exectest xset ; then + if ! timeout 1s xset q &>/dev/null ; then + NO_DISPLAY=1 + fi +elif [ -z "${DISPLAY}" ] ; then + NO_DISPLAY=1 +fi + +if [ "${NO_DISPLAY}" -eq 1 ] ; then + $_opt_quiet || echo "W: Running without X server. Not all data will be collected." +fi - echo "Generating hardware information report using file $INFOFILE." - echo "This might take a few seconds/minutes. Please be patient..." +cd "${OUTDIR}" || exit 1 +( + if ! $_opt_quiet ; then + [ -n "$GENERATE_FILE" ] && echo "Output file: $OUTFILE" + [ -n "$GENERATE_DIRECTORY" ] && echo "Output directory: $OUTDIR" + echo + echo "This might take a few seconds/minutes. Please be patient..." + fi -# some sysinfo - date > date + # some sysinfo + date > ./date if [ -r /etc/grml_version ] ; then cat /etc/grml_version > grml_version fi if [ -r /etc/debian_version ] ; then cat /etc/debian_version > debian_version fi - uname -a > uname + uname -a > ./uname -# disks / devices + # inxi + exectest inxi && inxi -F -xx > ./inxi 2>./inxi.error + exectest inxi && inxi -FJ --admin > ./inxi_admin 2>./inxi_admin.error + + # disks / devices [ -f /proc/scsi/scsi ] && cat /proc/scsi/scsi > scsi - (lspci; lspci -n) | sort > lspci - [ -r /proc/bus/pnp ] && lspnp > lspnp - [ -r /proc/bus/usb ] && lsusb > lsusb + exectest lspci && lspci -nn > ./lspci + exectest lspci && lspci -vvnn > ./lspci_verbose cat /proc/partitions > partitions find /proc/ide/ -name geometry -exec grep . {} \; > proc_ide 2>/dev/null - df -h > df 2>/dev/null - for i in free lsmod mount lsdev ; do - exectest $i && $i > $i + df -h > ./df 2>/dev/null + for i in free lsmod mount lsdev lspnp ; do + exectest $i && $i > ./$i done -# proc stuff + if exectest lsusb ; then + lsusb > ./lsusb + lsusb -v > ./lsusb_verbose 2>./lsusb_verbose.error + fi + + swapon -s > ./swapon 2>./swapon.error + + # proc stuff for i in cpuinfo interrupts cmdline devices dma fb iomem ioports \ - mdstat meminfo modules mtrr pci version ; do + mdstat meminfo modules mtrr pci uptime version ; do [ -r /proc/$i ] && cat /proc/$i > proc_$i done - exectest sysdump && sysdump > sysdump 2>sysdump.error -# for sysnet in `find /proc/sys/net/*/ -type f | grep -v flush`; do cat $sysnet >> sysnet ; done -# log + if ! $_opt_quiet ; then + echo "Starting sysdump..." + echo " NOTE: if it seems to be hanging at this stage file a bug report with output of:" + echo " lsof -p \$(pgrep -f "\$\(which sysdump\)")" + fi + exectest sysdump && sysdump > ./sysdump 2>./sysdump.error + if ! $_opt_quiet ; then + echo "Execution of sysdump finished." + fi + + exectest cpuid && cpuid > ./cpuid 2>./cpuid.error + + exectest uptime && uptime > ./uptime 2>./uptime.error + + # log dmesg > dmesg.cur -# hwinfo - exectest discover && discover -v --type-summary --enable-bus all > discover 2> discover.2 + # hwinfo + exectest discover && discover -v --type-summary --enable-bus all > ./discover 2>./discover.error exectest hwinfo && hwinfo log=hwinfo - exectest x86info && x86info > x86info 2>x86info.2 + exectest numactl && numactl --hardware > ./numactl + exectest x86info && x86info > ./x86info 2>./x86info.error + exectest lscpu && lscpu > ./lscpu + exectest lscpu && lscpu -e > ./lscpu_extended -# net stuff - exectest ifconfig && ifconfig -v -a > ifconfig - exectest ip && ip route show > ip_route - exectest ip && ip link show > ip_link - exectest route && route -n > route + # EFI + exectest efibootmgr && efibootmgr -v >efibootmgr 2>efibootmgr.error -# software + # net stuff, net-tools: + exectest ifconfig && ifconfig -v -a > ./ifconfig + exectest route && route -n > ./route + + # net stuff, ethtool + if exectest ethtool ; then + get_network_devices + for dev in "${niclist[@]}" ; do + ethtool "${dev}" > ethtool_"${dev}" + case "${dev}" in + "lo") + # skip the loopback device, `ethtool --driver lo` fails with: + # "Cannot get driver information: Operation not supported" + ;; + *) + ethtool --driver "${dev}" > ethtool_"${dev}_driver" + ;; + esac + done + fi + + # net stuff, iproute: + exectest ip && ip addrlabel list > ip_addrlabel + exectest ip && ip addr show > ip_addr + exectest ip && ip link show > ip_link + exectest ip && ip maddr show > ip_maddr + exectest ip && ip mroute show > ip_mroute + exectest ip && ip mrule show > ip_mrule 2>ip_mrule.error + exectest ip && ip neigh show > ip_neigh + exectest ip && ip netconf > ip_netconf + exectest ip && ip netns list > ip_netns + exectest ip && ip ntable show > ip_ntable + exectest ip && ip route show > ip_route + exectest ip && if [ -r /etc/iproute2/rt_tables ] ; then + grep -v '^#' /etc/iproute2/rt_tables | while read table _ ; do + ip route show table "${table}" > "ip_route_table_${table}" 2> "ip_route_table_${table}".error + done + fi + exectest ip && ip rule show > ip_rule + exectest ip && ip tunnel show > ip_tunnel + exectest ip && ip tuntap show > ip_tuntap + + # software if exectest dpkg ; then - dpkg --get-selections > dpkg_get_selections - COLUMNS=300 dpkg --list > dpkg_list - COLUMNS=1000 dpkg -l linux-image-$UNAME | grep linux-image-$UNAME | tr -s ' ' > running_kernel + dpkg --get-selections > dpkg_get_selections + COLUMNS=300 dpkg --list > dpkg_list + fi + + # power management + exectest laptop-detect && laptop-detect >/dev/null 2>/dev/null && echo "0" > laptop_detected + if [ -r /proc/acpi/info ] ; then + cat /proc/acpi/info > acpi_info + fi + + if exectest acpi ; then + acpi > ./acpi 2>acpi.error + acpi --everything > ./acpi.everything 2>./acpi.everything.error + acpi -v > ./acpi.version fi + [ -r /proc/apm/ ] && apm > ./apm -# powermanagement - exectest laptop-detect && laptop-detect 1>/dev/null 2>/dev/null && echo "0" > laptop_detected - exectest acpi_available && acpi_available && cat /proc/acpi/info > acpi_info - exectest acpi && acpi > acpi 2> acpi.error && acpi -v > acpi.version - [ -r /proc/apm/ ] && apm > apm + if exectest mcelog ; then + mcelog > ./mcelog 2>./mcelog.error + fi -# kernel stuff + # kernel stuff if [ -r /proc/config.gz ] ; then - zcat /proc/config.gz > kernelconfig + zcat /proc/config.gz > kernelconfig else - [ -r /boot/config-$UNAME ] && cat /boot/config-$UNAME > kernelconfig + [ -r "/boot/config-${UNAME}" ] && cat "/boot/config-${UNAME}" > kernelconfig fi -# X stuff - if [ -n "${DISPLAY}" ] ; then - exectest xviddetect && xviddetect > xviddetect - exectest xvidtune && xvidtune -show > xdivtune - exectest xrandr && xrandr > xrandr - exectest xdpyinfo && xdpyinfo > xdpyinfo - X -version 1> x_version 2>&1 + exectest dpkg && COLUMNS=1000 dpkg -l "linux-image-${UNAME}" 2>running_kernel.error \ + | grep "linux-image-${UNAME}" | tr -s ' ' > running_kernel 2>>running_kernel.error + dpkg -S "/boot/vmlinuz-$(uname -r)" >> running_kernel 2>>running_kernel.error + + # X stuff + if [ "${NO_DISPLAY}" -eq 0 ] ; then + exectest xviddetect && xviddetect > ./xviddetect + exectest xvidtune && xvidtune -show > ./xdivtune + exectest xrandr && xrandr > ./xrandr + exectest xdpyinfo && xdpyinfo > ./xdpyinfo + X -version > x_version 2>&1 fi for i in Xorg.0.log Xorg.7.log Xorg.8.log XFree86.0.log XFree86.7.log XFree86.8.log dmesg ; do - cp /var/log/$i log_$i 2>/dev/null + cp /var/log/$i log_$i 2>/dev/null done + if [ -r "$HOME"/.local/share/xorg/Xorg.0.log ] ; then + cp "$HOME"/.local/share/xorg/Xorg.0.log user_Xorg.0.log + fi + cp /etc/X11/xorg.conf xorg.conf 2>/dev/null cp /etc/modules modules 2>/dev/null cp /etc/X11/XF86Config-4 XF86Config-4 2>/dev/null -# not available: -# sysinfo -class device -level all + # as root: + if [ -n "$NOTROOT" ] ; then + echo "not running as root" > root + else + echo "running as root" > root + disk_info -# as root: -if [ -n "$NOTROOT" ] ; then - echo "not running as root" > root -else - echo "running as root" > root - disk_info - exectest sfdisk && sfdisk -d > sfdisk 2>sfdisk.error - exectest ddcprobe && ddcprobe > ddcprobe - exectest dmidecode && dmidecode > dmidecode + exectest dmidecode && dmidecode > ./dmidecode - exectest dconf && dconf -o dconf + if exectest mcelog ; then + mcelog --dmi > mcelog_dmi 2>mcelog_dmi.error + fi + + if exectest edac-util ; then + edac-util > ./edac-util 2>./edac-util.error + edac-util --report=full > edac-util_report 2>edac-util_report.error + fi - if [ -x /usr/share/doc/lm-sensors/examples/eeprom/decode-dimms.pl ] ; then + if exectest decode-dimms ; then + decode-dimms > ./decode-dimms 2>./decode-dimms.error + elif [ -x /usr/share/doc/lm-sensors/examples/eeprom/decode-dimms.pl ] ; then /usr/share/doc/lm-sensors/examples/eeprom/decode-dimms.pl > decode-dimms 2>decode-dimms.error - fi - - for disk in $disklist; do - exectest smartctl && smartctl -i /dev/$disk >> smartctl - exectest hdparm && hdparm -iv /dev/$disk >> hdparm - exectest fdisk && fdisk -lu /dev/$disk >> fdisk 2>>fdisk.error - exectest parted && parted -s /dev/$disk print >> parted - LC_ALL=C file -s /dev/$disk?* | grep -v ": empty" >> file_disk - done -fi + fi + + if exectest acpidump ; then + acpidump > ./acpidump 2>./acpidump.error + fi + + if exectest mokutil ; then + mokutil --sb-state > ./mokutil_state 2>./mokutil_state.error + fi + + # proxmox + exectest qm && qm list > ./qm 2>./qm.error + # libvirt + exectest virsh && virsh list >./virsh 2>./virsh.error + # openvz + exectest vzlist && vzlist >./vzlist 2>./vzlist.error + # vserver + exectest vserver-stat && vserver-stat >./vserver-stat 2>./vserver-stat.error + + exectest mdadm && mdadm --detail /dev/md[0-9]* >> ./mdadm 2>./mdadm.error + + # LVM + exectest pvs && pvs > ./pvs 2>./pvs.error + exectest vgs && vgs > ./vgs 2>./vgs.error + exectest lvs && lvs > ./lvs 2>./lvs.error + exectest lvdisplay && lvdisplay > ./lvdisplay 2>./lvdisplay.error + + exectest dmsetup && dmsetup ls > dmsetup_ls 2>dmsetup_ls.error + exectest dmsetup && dmsetup ls --tree > dmsetup_ls_tree 2>dmsetup_ls_tree.error + exectest lsblk && lsblk > ./lsblk 2>./lsblk.error + + # iSCSI + if exectest iscsiadm ; then + iscsiadm -m session > iscsiadm_session 2>iscsiadm_session.error + iscsiadm -m fw > iscsiadm_fw 2>iscsiadm_fw.error + iscsiadm -m host > iscsiadm_host 2>iscsiadm_host.error + iscsiadm -m iface > iscsiadm_iface 2>iscsiadm_iface.error + iscsiadm -m node > iscsiadm_node 2>iscsiadm_node.error + iscsiadm -m discovery > iscsiadm_discovery 2>iscsiadm_discovery.error + fi + + if exectest lsscsi ; then + lsscsi > ./lsscsi 2>./lsscsi.error + lsscsi -t > ./lsscsi_transport 2>./lsscsi_transport.error + fi + + for disk in $disklist; do + if exectest sfdisk && [[ -b "/dev/${disk}" ]] ; then + sfdisk -d "/dev/${disk}" > "./sfdisk_${disk}" 2>"./sfdisk_${disk}.error" + fi + + if exectest smartctl ; then + echo -e "smartctl -a /dev/${disk}:\n" >> smartctl + smartctl -a "/dev/$disk" >> ./smartctl + echo -e "\n\n" >> ./smartctl + fi + + if exectest hdparm ; then + echo -e "hdparm -iv /dev/${disk}:\n" >> hdparm + hdparm -iv "/dev/$disk" >> ./hdparm 2>> ./hdparm.error + echo -e "\n\n" >> hdparm + fi + + if exectest fdisk ; then + echo -e "fdisk -lu /dev/${disk}:\n" >> fdisk + fdisk -lu "/dev/$disk" >> ./fdisk 2>> ./fdisk.error + echo -e "\n\n" >> fdisk + fi + + if exectest parted ; then + echo -e "parted -s /dev/${disk}:\n" >> parted + parted -s "/dev/$disk" print >> ./parted + echo -e "\n\n" >> parted + fi + + if exectest sdparm ; then + echo -e "sdparm --all --long /dev/${disk}:\n" >> sdparm + sdparm --all --long "/dev/$disk" >> ./sdparm + echo -e "\n\n" >> sdparm + fi + + if exectest sg_inq ; then + echo -e "sg_inq /dev/${disk}:\n" >> sg_inq + sg_inq "/dev/$disk" >> ./sg_inq 2>> ./sg_inq.error + echo -e "\n\n" >> sg_inq + fi + + file -s "/dev/${disk}"?* | grep -v ": empty" >> file_disk + done + fi ) -# finally create the tarball -if [ -f "$INFOFILE" ] ; then - DATE="$(date +%Y%m%d_%H%M%S)" - NEW_INFOFILE="info_$DATE.tar.bz2" - echo - echo "Warning: $INFOFILE exists already, using $NEW_INFOFILE instead.">&2 - tar jcf $NEW_INFOFILE info/ && rm -Rf info/ - INFOFILE="$NEW_INFOFILE" # adjust variable for final info message +# get rid of empty files +for file in *.error ; do + test -s "$file" || rm -- "$file" +done + +$_opt_quiet || echo + +cd "${WORKING_DIR}" + +# create tarball +if [ -n "$GENERATE_FILE" ] ; then + tar acf "${OUTFILE}" "${OUTDIRNAME}" + if ! $_opt_quiet ; then + # shellcheck disable=SC2012 + [ -r "$OUTFILE" ] && echo "$OUTFILE ($(ls -ahl -- "$OUTFILE" | awk '{print $5}')) has been generated." + fi +fi + +# remove (temporary) output directory if needed, else keep it, as it doubles +# as the real output directory. +if [ -z "$GENERATE_DIRECTORY" ] ; then + rm -r "${OUTDIR}" else - tar jcf $INFOFILE info/ && rm -Rf info/ + if ! $_opt_quiet ; then + [ -r "${OUTDIR}" ] && echo "${OUTDIR} has been generated." + fi fi -echo -[ -r "$INFOFILE" ] && echo "$INFOFILE ($(ls -ahl $INFOFILE | awk '{print $5}')) has been generated." -echo +exit 0 ## END OF FILE##################################################################