Rev 100 | Go to most recent revision | Blame | Last modification | View Log | Download | RSS feed
#!/bin/ash## Functions library :: for Linux Live scripts 5.x.y# Author: Tomas M. <http://www.linux-live.org>## modified by Urs Beyerle, PSI# - to allow LiveCD mounted over nfs# - add scsi_mod, sd_mod for usb-storage module# - only with boot option "automount", all devices in fstab are rw automounted# - remove detect of CD and Floppy (done by fstab-sync)# - add sr_mod (USB CDROM support)# - add SATA to modprobe_usb_modules -> modprobe_usb_sata_modules# - add fscache (for SL5)# - add ide-cd, sr_mod, cdrom (for SL5 cdrom support)# - add aufs (unionfs replacement)# - to allow LiveCD mounted over NFS (for diskless client)# - add functions get_dhcp_lease() and load_network_modules()# - add ata_piix to modprobe_usb_sata_modules# - works with unionfs 2.x# - better detection of network card in case of diskless client## ===========================================================# user interface functions# ===========================================================# echolog# $1 = text to show and to write to /var/log/messages#echolog(){echo "LIVECD:" "$@" >>/var/log/livedbgecho "$@"}# debug# commands executed when debug boot parameter is present#debug(){echoecho "====="echo ": Debugging started. Here is the root shell for you."echo ": Type your desired command or hit Ctrl+D to continue booting."echoash}# header# $1 = text to show#header(){echolog "[0;1m$1[0;0m"}fatal(){header "Fatal error occured - $1"echolog "Something went wrong and we can't continue booting :("echolog "You may explore the system by using simple commands like ls, lsmod, mount, etc."echolog "You may also try to hit Ctrl+D. Booting will continue. Use at your own risk."echolog "To be safe, hit Ctrl+Alt+Delete to reboot."echologash}# ===========================================================# text processing functions# ===========================================================# egrep_o is a replacement for "egrep -o". It prints only the last# matching text# $1 = regular expression#egrep_o(){cat | egrep "$1" | sed -r "s/.*($1).*/\\1/"}# look into cmdline and echo $1 back if $1 is set# $1 = value name, case sensitive, for example livecd_subdir# $2 = file to use instead /proc/cmdline, optional#cmdline_parameter(){CMDLINE=/proc/cmdlineif [ "$2" != "" ]; then CMDLINE="$2"; ficat "$CMDLINE" | egrep_o "(^|[[:space:]]+)$1(\$|=|[[:space:]]+)" | egrep_o "$1"}# look into cmdline and echo value of $1 option# $1 = value name, case sensitive, for example livecd_subdir# $2 = file to use instead /proc/cmdline, optional#cmdline_value(){CMDLINE=/proc/cmdlineif [ "$2" != "" ]; then CMDLINE="$2"; ficat "$CMDLINE" | egrep_o "(^|[[:space:]]+)$1=([^[:space:]]+)" | egrep_o "=.*" | cut -b 2- | tail -n 1}# ===========================================================# system functions# ===========================================================# modprobe module $1, including all dependencies, suppress all messages# (own function because modprobe in busybox doesn't work with gzipped modules)# $1 = module name, eg. ehci-hcd# $2 = optional argument#modprobe_module(){if [ "$1" = "" ]; then return 1; fiPRINTK=`cat /proc/sys/kernel/printk`echo "0" >/proc/sys/kernel/printkKERNEL="`uname -r`"; LSMOD=/tmp/lsmodMODULEDEPS="`cat /lib/modules/$KERNEL/modules.dep | egrep \"$1\\.ko(\\.gz)?:\"`"for MODULE in `echo $MODULEDEPS | cut -d ":" -f 2-` `echo $MODULEDEPS | cut -d ":" -f 1`; doMODULE=${MODULE%%:}; # remove : at the end, a bugTMPMOD="/tmp/`basename $MODULE .gz`";# if the module is not loaded alreadyif [ "`cat $LSMOD 2>/dev/null | egrep \"^$TMPMOD\\\$\"`" = "" ]; thengunzip -c $MODULE 2>/dev/null >$TMPMODif [ $? -ne 0 ]; then cp $MODULE $TMPMOD; fi # can't gunzip? copyinsmod $TMPMOD $2; err=$?### insmod $TMPMOD $2 >/dev/null 2>/dev/null; err=$?if [ "$err" -eq 0 ]; then echo $TMPMOD >>$LSMOD; fi # module logrm $TMPMODfidoneecho "$PRINTK" >/proc/sys/kernel/printkif [ "$err" -ne 0 ]; then echolog "error inserting module $1 ($err)"; fireturn $err}# Mount device $1 to $2# $1 = /dev device to mount, eg. /dev/hda1# $2 = mountpoint, eg. /mnt/hda1# $3 = mount options, for example "loop", "ro", or "remount,rw"#mount_device(){mkdir -p $2if [ "$3" != "" ]; then OPTIONS="-o $3"; else OPTIONS=""; fiPRINTK=`cat /proc/sys/kernel/printk`echo "0" >/proc/sys/kernel/printkmount -t auto $1 $2 $OPTIONS >/dev/null 2>/dev/nullerr=$?if [ "$err" -ne 0 ]; then rmdir $2 2>/dev/null; fiecho "$PRINTK" >/proc/sys/kernel/printkreturn $err}# ===========================================================# live module functions# ===========================================================# Create module# call mksquashfs with apropriate arguments# $1 = directory which will be compressed to squashfs module# $2 = output .mo file# $3 = optional -keep-as-directory argument#create_module(){mksquashfs $1 $2 $3 >/dev/nullif [ $? -ne 0 ]; then return 1; fichmod oga-x $2 # remove execute attrib}# Mount .mo module to destination directory# $1 = path to .mo livecd compressed module# $2 = destination folder#mount_module(){mount -t squashfs -o loop,ro $1 $2}# Insert a directory tree $2 to an union specified by $1# Top-level read-write branch is specified by it's index 0# $1 = union absolute path (starting with /)# $2 = path to data directory#union_insert_dir(){which unionctl >/dev/null 2>&1if [ $? -eq 0 ]; then# unionfs 1.x or aufsunionctl "$1" --add --after 0 --mode ro "$2"else# unionfs 2.xmount -t unionfs -o remount,add=:${2}=ro none "$1"fi}# List all modules in all directories (base, modules, optional)# and filter out unneeded optional modules (not specified by load= kernel parameter)# $1 = root directory of mounted DATAdir#list_modules(){LOAD="`cmdline_value load`"ls -A1d $1/*.mo $1/*/*.mo 2>/dev/null | while read LINE; doMODNAME="`basename $LINE .mo`"if [ "$LOAD" != "*" -a "`echo $LINE | grep optional`" != "" -a "`echo $LOAD | egrep \"(^|,)$MODNAME(\\\$|,)\"`" = "" ]; then continueelse echo $LINE; fidone}# Insert one single .mo module to the union# $1 = union absolute path (starting with /)# $2 = module.mo full path# $3 = destination folder, where images will be mounted to#union_insert_module(){TARGET="$3/`basename $2`"while [ -e $TARGET ]; do TARGET=$TARGET.X; donemkdir -p $TARGETmount_module $2 $TARGETif [ $? -ne 0 ]; then echo "can't read module data"; return 1; fiunion_insert_dir $1 $TARGET;}# Insert all .mo modules, in $2 directory and subdirectories, to the union# $1 = union absolute path (starting with /)# $2 = LiveCD data dir (with directories /base, /modules, etc.)# $3 = destination folder, where images will be mounted to#union_insert_modules(){PRINTK=`cat /proc/sys/kernel/printk`echo "0" >/proc/sys/kernel/printklist_modules $2 | while read MODULE; doecho " -> `basename $MODULE`"union_insert_module $1 $MODULE $3doneecho "$PRINTK" >/proc/sys/kernel/printk}# Copy modules to RAM directory# $1 = data directory# $2 = target directory in RAM#copy_to_ram(){cp -R $1/* $2if [ $? -ne 0 ]; then fatal "can't copy to RAM, not enough memory?"; fi}# Copy content of "rootcopy" directory on the CD to $2 (union, usually)# $1 = source# $2 = destination#copy_rootchanges(){cp -a $1/rootcopy/* $2 2>/dev/null # could be empty}# ===========================================================# discovery functions# ===========================================================# List all CD-ROMs# by using /proc entries#list_cdrom_devices(){if [ "`cmdline_parameter nocd`" != "" ]; then return 1; fifor CDDEVICE in `cat /proc/sys/dev/cdrom/info | head -n 3 | tail -n 1 | cut -d ":" -f 2`; doecho "/dev/$CDDEVICE"done}# List all partition devices# take list of all partitions and output unique disks.# Return empty result when nohd parameter was given.#list_partition_devices(){if [ "`cmdline_parameter nohd`" != "" ]; then return 1; ficat /proc/partitions | grep -v loop | sed -r "s/^[0-9[:space:]]+/\/dev\//" | grep /dev/}# List all disk devices#list_disk_devices(){list_partition_devices | egrep -v "[0-9]"}# List all block devices#list_block_devices(){list_cdrom_deviceslist_partition_devices}# Try to mount all disks, partitions and cdroms and Find where the LiveCD is.# If LiveCD found in the device, echo dirname of it's directory,# and leave the device mounted. Mounting is not ro, but without any argument.# $1 = directory where devices will be mounted# added: mount $NFSROOT to /$1/nfs if NFSROOT is set. and search there for LiveCD#find_live_data_dir(){if [ "$NFSROOT" != "" ]; thenDIR="/$1/nfs"mkdir -p $DIRmount -t nfs -o nolock,ro,rsize=8192,wsize=8192,hard,intr $NFSROOT $DIRFOUND=`ls -A1d $DIR/livecd.sgn $DIR/*/livecd.sgn 2>/dev/null | head -n 1`if [ "$FOUND" != "" ]; thendirname "$FOUND"fielselist_block_devices | while read DEVICE; doDIR="/$1/`basename $DEVICE`"mount_device $DEVICE $DIRif [ $? -ne 0 ]; then continue; fiFOUND=`ls -A1d $DIR/livecd.sgn $DIR/*/livecd.sgn 2>/dev/null | head -n 1`if [ "$FOUND" = "" ]; then umount $DIR 2>/dev/null; rmdir $DIR 2>/dev/nullelse dirname "$FOUND"; return 1; fidonefi}# ===========================================================# hardware preparation functions# ===========================================================# Create block devices to /dev described by /sys entries#create_block_devices(){echolog "creating /dev entries for block devices"ls -A1d /sys/block/*/dev /sys/block/*/*/dev 2>/dev/null | grep -v loop | while read BLOCK; doDEVICE="/dev/`basename \`dirname $BLOCK\``"if [ ! -b $DEVICE ]; thenMINORMAJOR="`head -n 1 $BLOCK | tr ':' ' '`"mknod $DEVICE b $MINORMAJORfidone}# modprobe kernel modules needed for the LiveCD#modprobe_essential_modules(){echolog "starting loop device support"modprobe_module loop max_loop=32echolog "starting cdrom support"modprobe_module ide_cdmodprobe_module ide-cdmodprobe_module sr_modmodprobe_module cdromecholog "starting cdrom filesystem support"modprobe_module isofsecholog "starting squashfs support"modprobe_module squashfsecholog "starting unionfs/aufs support"modprobe_module unionfsmodprobe_module aufsecholog "starting vfat support"modprobe_module nls_cp437modprobe_module nls_iso8859-1modprobe_module nls_iso8859-2modprobe_module vfatecholog "starting ntfs support"modprobe_module ntfscreate_block_devices}# modprobe kernel modules needed for USB masstorage devices#modprobe_usb_sata_modules(){echolog "starting USB and SATA support"modprobe_module ehci-hcdmodprobe_module ohci-hcdmodprobe_module uhci-hcdmodprobe_module scsi_modmodprobe_module libatamodprobe_module ata_piixmodprobe_module ahcimodprobe_module sd_modmodprobe_module sr_modmodprobe_module usb-storageecholog "waiting for USB devices, max 9 seconds"sleep 9create_block_devices}# modprobe nfs kernel modules#modprobe_nfs_modules(){echolog "starting nfs support"modprobe_module lockdmodprobe_module fscachemodprobe_module nfs_aclmodprobe_module nfs}# enable/disable CD autoejecting when unmounted# $1 = 1|0 ... enable|disable#cd_autoeject(){echo $1 >/proc/sys/dev/cdrom/autoeject}# Disable DMA if nodma boot parameter is present#setup_dma(){if [ ! "`cmdline_parameter nodma`" = "" ]; thenfor DEVICE in `list_cdrom_devices` `list_disk_devices`; doecholog "setting DMA support off for $DEVICE"hdparm -d 0 $DEVICEdonefi}# create correct fstab file in $1/etc/fstab and create apropriate# mount directories in $1# $1 = root directory (union)#activate_fstab(){mkdir -p $1/etcFSTAB="$1/etc/fstab"echo "tmpfs / tmpfs defaults 0 0" >$FSTABecho "devpts /dev/pts devpts gid=5,mode=620 0 0" >>$FSTABecho "proc /proc proc defaults 0 0" >>$FSTAB# all the rest will be done by runlast or by fstab-sync# search for SWAP done later in /etc/rc.sysinit}# ===========================================================# functions used for LiveCD on NFS# ===========================================================# load network modules, if NFSROOT is set# Urs Beyerle, PSI#load_network_modules(){# mii maybe need by NICecholog "load module mii"modprobe_module miiFOUND_NIC=""# 1. load network driver defined by kernel parameter nicNIC="`cmdline_value nic`"if [ -n "$NIC" ]; thenecholog "load module $NIC"modprobe_module nicfi# 2. detecting network card from pcitable listPCITABLE=/bin/pcitableNICS=`lspci -n | awk '/Class 0200/ {print $4}' | tr ':' ' ' \| while read x y ; do grep "0x$x.*0x$y" $PCITABLE \| awk '$3 !~ /"unknown"/ {print $3}' | tr -d '"' ; done`if [ -n "$NICS" ]; thenecholog "found network card(s): $NICS"for nic in $NICS; doecholog "load module $nic"modprobe_module $nicFOUND_NIC="yes"donefi# 3. probe for more network card drivers, if no ethX devices foundfor iface in eth0 eth1 eth2 eth3; doifconfig $iface > /dev/null 2>&1if [ $? -eq 0 ]; then FOUND_NIC="yes"; break; fidoneif [ -z $FOUND_NIC ]; thenKERNEL="`uname -r`"NETDRIVERS="`cat /lib/modules/$KERNEL/netdrivers`"for driver in $NETDRIVERS; domodprobe_module $driverifconfig $iface > /dev/null 2>&1if [ $? -eq 0 ]; thenecholog "load $driver"FOUND_NIC="yes"breakfirmmod $driverdonefi# 4. ask the user for a network driverif [ -z $FOUND_NIC ]; thenecho "ERROR: No network card detected!"echo "Type in your network card driver (e.g. tg3, e1000). "echo "Or ask your system administrator for help."echo -n "Network card driver: "read nicecho "Load module $nic"modprobe_module $nicfi}# get DHCP lease# Urs Beyerle, PSI#get_dhcp_lease(){# create /dev/urandom (needed by udhcpc)mknod /dev/urandom c 1 9for iface in eth0 eth0 eth1 eth1; doecholog "Try to get DHCP lease on $iface"udhcpc --now --quit --interface=$iface --script=/bin/udhcpc.script[ $? -eq 0 ] && returnecho "ERROR: couldn't get DHCP lease, trying again"doneecho "ERROR: can't get DHCP lease on eth0 and eth1"}