#! /bin/sh # # Linux Additions kernel module init script ($Revision: 43742 $) # # # Copyright (C) 2006-2010 Oracle Corporation # # This file is part of VirtualBox Open Source Edition (OSE), as # available from http://www.virtualbox.org. This file is free software; # you can redistribute it and/or modify it under the terms of the GNU # General Public License (GPL) as published by the Free Software # Foundation, in version 2 as it comes in the "COPYING" file of the # VirtualBox OSE distribution. VirtualBox OSE is distributed in the # hope that it will be useful, but WITHOUT ANY WARRANTY of any kind. # # chkconfig: 357 30 70 # description: VirtualBox Linux Additions kernel modules # ### BEGIN INIT INFO # Provides: vboxadd # Required-Start: # Required-Stop: # Default-Start: 2 3 4 5 # Default-Stop: 0 1 6 # Description: VirtualBox Linux Additions kernel modules ### END INIT INFO PATH=$PATH:/bin:/sbin:/usr/sbin PACKAGE=VBoxGuestAdditions LOG="/var/log/vboxadd-install.log" MODPROBE=/sbin/modprobe OLDMODULES="vboxguest vboxadd vboxsf vboxvfs vboxvideo" if $MODPROBE -c | grep -q '^allow_unsupported_modules *0'; then MODPROBE="$MODPROBE --allow-unsupported-modules" fi # Check architecture cpu=`uname -m`; case "$cpu" in i[3456789]86|x86) cpu="x86" lib_path="/usr/lib" ;; x86_64|amd64) cpu="amd64" if test -d "/usr/lib64"; then lib_path="/usr/lib64" else lib_path="/usr/lib" fi ;; esac if [ -f /etc/arch-release ]; then system=arch elif [ -f /etc/redhat-release ]; then system=redhat elif [ -f /etc/SuSE-release ]; then system=suse elif [ -f /etc/gentoo-release ]; then system=gentoo elif [ -f /etc/lfs-release -a -d /etc/rc.d/init.d ]; then system=lfs else system=other fi if [ "$system" = "arch" ]; then USECOLOR=yes . /etc/rc.d/functions fail_msg() { stat_fail } succ_msg() { stat_done } begin() { stat_busy "$1" } fi if [ "$system" = "redhat" ]; then . /etc/init.d/functions fail_msg() { echo_failure echo } succ_msg() { echo_success echo } begin() { echo -n "$1" } fi if [ "$system" = "suse" ]; then . /etc/rc.status fail_msg() { rc_failed 1 rc_status -v } succ_msg() { rc_reset rc_status -v } begin() { echo -n "$1" } fi if [ "$system" = "gentoo" ]; then if [ -f /sbin/functions.sh ]; then . /sbin/functions.sh elif [ -f /etc/init.d/functions.sh ]; then . /etc/init.d/functions.sh fi fail_msg() { eend 1 } succ_msg() { eend $? } begin() { ebegin $1 } if [ "`which $0`" = "/sbin/rc" ]; then shift fi fi if [ "$system" = "lfs" ]; then . /etc/rc.d/init.d/functions fail_msg() { echo_failure } succ_msg() { echo_ok } begin() { echo $1 } fi if [ "$system" = "other" ]; then fail_msg() { echo " ...fail!" } succ_msg() { echo " ...done." } begin() { echo -n $1 } fi show_error() { if [ "$system" = "gentoo" ]; then eerror $1 fi fail_msg echo "($1)" } fail() { show_error "$1" exit 1 } dev=/dev/vboxguest userdev=/dev/vboxuser config=/var/lib/VBoxGuestAdditions/config owner=vboxadd group=1 test_for_gcc_and_make() { which make > /dev/null 2>&1 || printf "\nThe make utility was not found. If the following module compilation fails then\nthis could be the reason and you should try installing it.\n" which gcc > /dev/null 2>&1 || printf "\nThe gcc utility was not found. If the following module compilation fails then\nthis could be the reason and you should try installing it.\n" } test_sane_kernel_dir() { KERN_VER=`uname -r` KERN_DIR="/lib/modules/$KERN_VER/build" if [ -d "$KERN_DIR" ]; then KERN_REL=`make -sC $KERN_DIR --no-print-directory kernelrelease 2>/dev/null || true` if [ -z "$KERN_REL" -o "x$KERN_REL" = "x$KERN_VER" ]; then return 0 fi fi printf "\nThe headers for the current running kernel were not found. If the following\nmodule compilation fails then this could be the reason.\n" if [ "$system" = "redhat" ]; then if echo "$KERN_VER" | grep -q "uek"; then printf "The missing package can be probably installed with\nyum install kernel-uek-devel-$KERN_VER\n" else printf "The missing package can be probably installed with\nyum install kernel-devel-$KERN_VER\n" fi elif [ "$system" = "suse" ]; then KERN_VER_SUSE=`echo "$KERN_VER" | sed 's/.*-\([^-]*\)/\1/g'` KERN_VER_BASE=`echo "$KERN_VER" | sed 's/\(.*\)-[^-]*/\1/g'` printf "The missing package can be probably installed with\nzypper install kernel-$KERN_VER_SUSE-devel-$KERN_VER_BASE\n" elif [ "$system" = "debian" ]; then printf "The missing package can be probably installed with\napt-get install linux-headers-$KERN_VER\n" fi } running_vboxguest() { lsmod | grep -q "vboxguest[^_-]" } running_vboxadd() { lsmod | grep -q "vboxadd[^_-]" } running_vboxsf() { lsmod | grep -q "vboxsf[^_-]" } running_vboxvideo() { lsmod | grep -q "vboxvideo[^_-]" } do_vboxguest_non_udev() { if [ ! -c $dev ]; then maj=`sed -n 's;\([0-9]\+\) vboxguest;\1;p' /proc/devices` if [ ! -z "$maj" ]; then min=0 else min=`sed -n 's;\([0-9]\+\) vboxguest;\1;p' /proc/misc` if [ ! -z "$min" ]; then maj=10 fi fi test -z "$maj" && { rmmod vboxguest 2>/dev/null fail "Cannot locate the VirtualBox device" } mknod -m 0664 $dev c $maj $min || { rmmod vboxguest 2>/dev/null fail "Cannot create device $dev with major $maj and minor $min" } fi chown $owner:$group $dev 2>/dev/null || { rm -f $dev 2>/dev/null rm -f $userdev 2>/dev/null rmmod vboxguest 2>/dev/null fail "Cannot change owner $owner:$group for device $dev" } if [ ! -c $userdev ]; then maj=10 min=`sed -n 's;\([0-9]\+\) vboxuser;\1;p' /proc/misc` if [ ! -z "$min" ]; then mknod -m 0666 $userdev c $maj $min || { rm -f $dev 2>/dev/null rmmod vboxguest 2>/dev/null fail "Cannot create device $userdev with major $maj and minor $min" } chown $owner:$group $userdev 2>/dev/null || { rm -f $dev 2>/dev/null rm -f $userdev 2>/dev/null rmmod vboxguest 2>/dev/null fail "Cannot change owner $owner:$group for device $userdev" } fi fi } start() { begin "Starting the VirtualBox Guest Additions "; uname -r | grep -q '^2\.6' 2>/dev/null && ps -A -o comm | grep -q '/*udevd$' 2>/dev/null || no_udev=1 running_vboxguest || { rm -f $dev || { fail "Cannot remove $dev" } rm -f $userdev || { fail "Cannot remove $userdev" } $MODPROBE vboxguest >/dev/null 2>&1 || { fail "modprobe vboxguest failed" } case "$no_udev" in 1) sleep .5;; esac } case "$no_udev" in 1) do_vboxguest_non_udev;; esac running_vboxsf || { $MODPROBE vboxsf > /dev/null 2>&1 || { if dmesg | grep "vboxConnect failed" > /dev/null 2>&1; then fail_msg echo "Unable to start shared folders support. Make sure that your VirtualBox build" echo "supports this feature." exit 1 fi fail "modprobe vboxsf failed" } } # This is needed as X.Org Server 1.13 does not auto-load the module. running_vboxvideo || $MODPROBE vboxvideo > /dev/null 2>&1 # Mount all shared folders from /etc/fstab. Normally this is done by some # other startup script but this requires the vboxdrv kernel module loaded. # This isn't necessary anymore as the vboxsf module is autoloaded. # mount -a -t vboxsf succ_msg return 0 } stop() { begin "Stopping VirtualBox Additions "; if ! umount -a -t vboxsf 2>/dev/null; then fail "Cannot unmount vboxsf folders" fi if running_vboxsf; then rmmod vboxsf 2>/dev/null || fail "Cannot unload module vboxsf" fi if running_vboxguest; then rmmod vboxguest 2>/dev/null || fail "Cannot unload module vboxguest" rm -f $userdev || fail "Cannot unlink $userdev" rm -f $dev || fail "Cannot unlink $dev" fi succ_msg return 0 } restart() { stop && start return 0 } # Remove any existing VirtualBox guest kernel modules from the disk, but not # from the kernel as they may still be in use cleanup_modules() { begin "Removing existing VirtualBox DKMS kernel modules" $DODKMS uninstall $OLDMODULES > $LOG succ_msg begin "Removing existing VirtualBox non-DKMS kernel modules" for i in $OLDMODULES; do find /lib/modules -name $i\* | xargs rm 2>/dev/null done succ_msg } # Build and install the VirtualBox guest kernel modules setup_modules() { # don't stop the old modules here -- they might be in use cleanup_modules begin "Building the VirtualBox Guest Additions kernel modules" # Short cut out if a dkms build succeeds if $DODKMS install vboxguest $INSTALL_VER >> $LOG 2>&1; then succ_msg return 0 fi test_for_gcc_and_make test_sane_kernel_dir echo begin "Building the main Guest Additions module" if ! $BUILDINTMP \ --save-module-symvers /tmp/vboxguest-Module.symvers \ --module-source $MODULE_SRC/vboxguest \ --no-print-directory install >> $LOG 2>&1; then show_error "Look at $LOG to find out what went wrong" return 1 fi succ_msg begin "Building the shared folder support module" if ! $BUILDINTMP \ --use-module-symvers /tmp/vboxguest-Module.symvers \ --module-source $MODULE_SRC/vboxsf \ --no-print-directory install >> $LOG 2>&1; then show_error "Look at $LOG to find out what went wrong" return 1 fi succ_msg if expr `uname -r` '<' '2.6.27' > /dev/null; then echo "Not building the VirtualBox advanced graphics driver as this Linux version is" echo "too old to use it." else begin "Building the OpenGL support module" if ! $BUILDINTMP \ --use-module-symvers /tmp/vboxguest-Module.symvers \ --module-source $MODULE_SRC/vboxvideo \ --no-print-directory install >> $LOG 2>&1; then show_error "Look at $LOG to find out what went wrong" return 1 fi succ_msg fi depmod return 0 } # Do non-kernel bits needed for the kernel modules to work properly (user # creation, udev, mount helper...) extra_setup() { begin "Doing non-kernel setup of the Guest Additions" echo "Creating user for the Guest Additions." >> $LOG # This is the LSB version of useradd and should work on recent # distributions useradd -d /var/run/vboxadd -g 1 -r -s /bin/false vboxadd >/dev/null 2>&1 # And for the others, we choose a UID ourselves useradd -d /var/run/vboxadd -g 1 -u 501 -o -s /bin/false vboxadd >/dev/null 2>&1 # Add a group "vboxsf" for Shared Folders access # All users which want to access the auto-mounted Shared Folders have to # be added to this group. groupadd -f vboxsf >/dev/null 2>&1 # Create udev description file if [ -d /etc/udev/rules.d ]; then echo "Creating udev rule for the Guest Additions kernel module." >> $LOG udev_call="" udev_app=`which udevadm 2> /dev/null` if [ $? -eq 0 ]; then udev_call="${udev_app} version 2> /dev/null" else udev_app=`which udevinfo 2> /dev/null` if [ $? -eq 0 ]; then udev_call="${udev_app} -V 2> /dev/null" fi fi udev_fix="=" if [ "${udev_call}" != "" ]; then udev_out=`${udev_call}` udev_ver=`expr "$udev_out" : '[^0-9]*\([0-9]*\)'` if [ "$udev_ver" = "" -o "$udev_ver" -lt 55 ]; then udev_fix="" fi fi ## @todo 60-vboxadd.rules -> 60-vboxguest.rules ? echo "KERNEL=${udev_fix}\"vboxguest\", NAME=\"vboxguest\", OWNER=\"vboxadd\", MODE=\"0660\"" > /etc/udev/rules.d/60-vboxadd.rules echo "KERNEL=${udev_fix}\"vboxuser\", NAME=\"vboxuser\", OWNER=\"vboxadd\", MODE=\"0666\"" >> /etc/udev/rules.d/60-vboxadd.rules fi # Put mount.vboxsf in the right place ln -sf "$lib_path/$PACKAGE/mount.vboxsf" /sbin # At least Fedora 11 and Fedora 12 require the correct security context when # executing this command from service scripts. Shouldn't hurt for other # distributions. chcon -u system_u -t mount_exec_t "$lib_path/$PACKAGE/mount.vboxsf" > /dev/null 2>&1 # And at least Fedora 15 needs this for the acceleration support check to # work redhat_release=`cat /etc/redhat-release 2> /dev/null` case "$redhat_release" in Fedora\ release\ 15* ) for i in "$lib_path"/*.so do restorecon "$i" >/dev/null done ;; esac succ_msg } # setup_script setup() { if test -r $config; then . $config else fail "Configuration file $config not found" fi test -n "$INSTALL_DIR" -a -n "$INSTALL_VER" || fail "Configuration file $config not complete" export BUILD_TYPE export USERNAME MODULE_SRC="$INSTALL_DIR/src/vboxguest-$INSTALL_VER" BUILDINTMP="$MODULE_SRC/build_in_tmp" DODKMS="$MODULE_SRC/do_dkms" chcon -t bin_t "$BUILDINTMP" > /dev/null 2>&1 chcon -t bin_t "$DODKMS" > /dev/null 2>&1 setup_modules mod_succ="$?" extra_setup if [ "$mod_succ" -eq "0" ]; then if running_vboxguest || running_vboxadd; then printf "You should restart your guest to make sure the new modules are actually used\n\n" else start fi fi } # cleanup_script cleanup() { if test -r $config; then . $config test -n "$INSTALL_DIR" -a -n "$INSTALL_VER" || fail "Configuration file $config not complete" DODKMS="$INSTALL_DIR/src/vboxguest-$INSTALL_VER/do_dkms" elif test -x ./do_dkms; then # Executing as part of the installer... DODKMS=./do_dkms else fail "Configuration file $config not found" fi # Delete old versions of VBox modules. cleanup_modules depmod # Remove old module sources for i in $OLDMODULES; do rm -rf /usr/src/$i-* done # Remove other files rm /sbin/mount.vboxsf 2>/dev/null rm /etc/udev/rules.d/60-vboxadd.rules 2>/dev/null } dmnstatus() { if running_vboxguest; then echo "The VirtualBox Additions are currently running." else echo "The VirtualBox Additions are not currently running." fi } case "$1" in start) start ;; stop) stop ;; restart) restart ;; setup) setup ;; cleanup) cleanup ;; status) dmnstatus ;; *) echo "Usage: $0 {start|stop|restart|status|setup}" exit 1 esac exit